index.js 2.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130
  1. /**
  2. * jsproxy cfworker api
  3. *
  4. * @update: 2019-05-03
  5. * @author: EtherDream
  6. * @see: https://github.com/EtherDream/jsproxy/
  7. */
  8. const pairs = Object.entries
  9. addEventListener('fetch', e => {
  10. const ret = handler(e.request)
  11. .catch(err => new Response(err))
  12. e.respondWith(ret)
  13. })
  14. const PREFLIGHT_INIT = {
  15. status: 204,
  16. headers: new Headers({
  17. 'access-control-allow-origin': '*',
  18. 'access-control-allow-methods': 'GET,POST,PUT,PATCH,TRACE,DELETE,HEAD,OPTIONS',
  19. 'access-control-allow-headers': '--url,--referer,--cookie,--origin,--ext,--aceh,--ver,--type,--mode,accept,accept-charset,accept-encoding,accept-language,accept-datetime,authorization,cache-control,content-length,content-type,date,if-match,if-modified-since,if-none-match,if-range,if-unmodified-since,max-forwards,pragma,range,te,upgrade,upgrade-insecure-requests,x-requested-with,chrome-proxy',
  20. 'access-control-max-age': '1728000',
  21. }),
  22. }
  23. /**
  24. * @param {Request} req
  25. */
  26. async function handler(req) {
  27. const reqHdrRaw = req.headers
  28. // preflight
  29. if (reqHdrRaw.has('access-control-request-headers')) {
  30. return new Response(null, PREFLIGHT_INIT)
  31. }
  32. let url = ''
  33. let extHdrs = null
  34. let acehOld = false
  35. const reqHdrNew = new Headers(reqHdrRaw)
  36. for (const [k, v] of reqHdrRaw.entries()) {
  37. if (!k.startsWith('--')) {
  38. continue
  39. }
  40. const k2 = k.substr(2)
  41. switch (k2) {
  42. case 'url':
  43. url = v
  44. break
  45. case 'aceh':
  46. acehOld = true
  47. break
  48. case 'mode':
  49. case 'type':
  50. break
  51. case 'ext':
  52. extHdrs = JSON.parse(v)
  53. break
  54. default:
  55. reqHdrNew.set(k2, v)
  56. break
  57. }
  58. }
  59. if (extHdrs) {
  60. for (const [k, v] of pairs(extHdrs)) {
  61. reqHdrNew.set(k, v)
  62. }
  63. }
  64. // proxy
  65. const res = await fetch(url, {
  66. method: req.method,
  67. headers: reqHdrNew,
  68. })
  69. // header filter
  70. const resHdrOld = res.headers
  71. const resHdrNew = new Headers(resHdrOld)
  72. let expose = '*'
  73. let vary = '--url'
  74. for (const [k, v] of resHdrOld.entries()) {
  75. if (k === 'access-control-allow-origin' ||
  76. k === 'access-control-expose-headers' ||
  77. k === 'location' ||
  78. k === 'set-cookie'
  79. ) {
  80. const x = '--' + k
  81. resHdrNew.set(x, v)
  82. if (acehOld) {
  83. expose = expose + ',' + x
  84. }
  85. resHdrNew.delete(k)
  86. }
  87. else if (k === 'vary') {
  88. vary = vary + ',' + v
  89. }
  90. else if (acehOld &&
  91. k !== 'cache-control' &&
  92. k !== 'content-language' &&
  93. k !== 'content-type' &&
  94. k !== 'expires' &&
  95. k !== 'last-modified' &&
  96. k !== 'pragma'
  97. ) {
  98. expose = expose + ',' + k
  99. }
  100. }
  101. if (acehOld) {
  102. expose = expose + ',--s'
  103. resHdrNew.set('--t', '1')
  104. }
  105. resHdrNew.set('access-control-expose-headers', expose)
  106. resHdrNew.set('access-control-allow-origin', '*')
  107. resHdrNew.set('vary', vary)
  108. resHdrNew.set('--s', res.status)
  109. return new Response(res.body, {
  110. status: 200,
  111. headers: resHdrNew,
  112. })
  113. }