index.js 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290
  1. 'use strict'
  2. /**
  3. * static files (404.html, sw.js, conf.js)
  4. */
  5. const ASSET_URL = 'https://etherdream.github.io/jsproxy'
  6. const JS_VER = 10
  7. const MAX_RETRY = 1
  8. /** @type {RequestInit} */
  9. const PREFLIGHT_INIT = {
  10. status: 204,
  11. headers: new Headers({
  12. 'access-control-allow-origin': '*',
  13. 'access-control-allow-methods': 'GET,POST,PUT,PATCH,TRACE,DELETE,HEAD,OPTIONS',
  14. 'access-control-max-age': '1728000',
  15. }),
  16. }
  17. /**
  18. * @param {any} body
  19. * @param {number} status
  20. * @param {Object<string, string>} headers
  21. */
  22. function makeRes(body, status = 200, headers = {}) {
  23. headers['--ver'] = JS_VER
  24. headers['access-control-allow-origin'] = '*'
  25. return new Response(body, {status, headers})
  26. }
  27. /**
  28. * @param {string} urlStr
  29. */
  30. function newUrl(urlStr) {
  31. try {
  32. return new URL(urlStr)
  33. } catch (err) {
  34. return null
  35. }
  36. }
  37. addEventListener('fetch', e => {
  38. const ret = fetchHandler(e)
  39. .catch(err => makeRes('cfworker error:\n' + err.stack, 502))
  40. e.respondWith(ret)
  41. })
  42. /**
  43. * @param {FetchEvent} e
  44. */
  45. async function fetchHandler(e) {
  46. const req = e.request
  47. const urlStr = req.url
  48. const urlObj = new URL(urlStr)
  49. const path = urlObj.href.substr(urlObj.origin.length)
  50. if (urlObj.protocol === 'http:') {
  51. urlObj.protocol = 'https:'
  52. return makeRes('', 301, {
  53. 'strict-transport-security': 'max-age=99999999; includeSubDomains; preload',
  54. 'location': urlObj.href,
  55. })
  56. }
  57. if (path.startsWith('/http/')) {
  58. return httpHandler(req, path.substr(6))
  59. }
  60. switch (path) {
  61. case '/http':
  62. return makeRes('请更新 cfworker 到最新版本!')
  63. case '/ws':
  64. return makeRes('not support', 400)
  65. case '/works':
  66. return makeRes('it works')
  67. default:
  68. // static files
  69. return fetch(ASSET_URL + path)
  70. }
  71. }
  72. /**
  73. * @param {Request} req
  74. * @param {string} pathname
  75. */
  76. function httpHandler(req, pathname) {
  77. const reqHdrRaw = req.headers
  78. if (reqHdrRaw.has('x-jsproxy')) {
  79. return Response.error()
  80. }
  81. // preflight
  82. if (req.method === 'OPTIONS' &&
  83. reqHdrRaw.has('access-control-request-headers')
  84. ) {
  85. return new Response(null, PREFLIGHT_INIT)
  86. }
  87. let acehOld = false
  88. let rawSvr = ''
  89. let rawLen = ''
  90. let rawEtag = ''
  91. const reqHdrNew = new Headers(reqHdrRaw)
  92. reqHdrNew.set('x-jsproxy', '1')
  93. // 此处逻辑和 http-dec-req-hdr.lua 大致相同
  94. // https://github.com/EtherDream/jsproxy/blob/master/lua/http-dec-req-hdr.lua
  95. const refer = reqHdrNew.get('referer')
  96. const query = refer.substr(refer.indexOf('?') + 1)
  97. if (!query) {
  98. return makeRes('missing params', 403)
  99. }
  100. const param = new URLSearchParams(query)
  101. for (const [k, v] of Object.entries(param)) {
  102. if (k.substr(0, 2) === '--') {
  103. // 系统信息
  104. switch (k.substr(2)) {
  105. case 'aceh':
  106. acehOld = true
  107. break
  108. case 'raw-info':
  109. [rawSvr, rawLen, rawEtag] = v.split('|')
  110. break
  111. }
  112. } else {
  113. // 还原 HTTP 请求头
  114. if (v) {
  115. reqHdrNew.set(k, v)
  116. } else {
  117. reqHdrNew.delete(k)
  118. }
  119. }
  120. }
  121. if (!param.has('referer')) {
  122. reqHdrNew.delete('referer')
  123. }
  124. // cfworker 会把路径中的 `//` 合并成 `/`
  125. const urlStr = pathname.replace(/^(https?):\/+/, '$1://')
  126. const urlObj = newUrl(urlStr)
  127. if (!urlObj) {
  128. return makeRes('invalid proxy url: ' + urlStr, 403)
  129. }
  130. /** @type {RequestInit} */
  131. const reqInit = {
  132. method: req.method,
  133. headers: reqHdrNew,
  134. redirect: 'manual',
  135. }
  136. if (req.method === 'POST') {
  137. reqInit.body = req.body
  138. }
  139. return proxy(urlObj, reqInit, acehOld, rawLen, 0)
  140. }
  141. /**
  142. *
  143. * @param {URL} urlObj
  144. * @param {RequestInit} reqInit
  145. * @param {number} retryTimes
  146. */
  147. async function proxy(urlObj, reqInit, acehOld, rawLen, retryTimes) {
  148. const res = await fetch(urlObj.href, reqInit)
  149. const resHdrOld = res.headers
  150. const resHdrNew = new Headers(resHdrOld)
  151. let expose = '*'
  152. for (const [k, v] of resHdrOld.entries()) {
  153. if (k === 'access-control-allow-origin' ||
  154. k === 'access-control-expose-headers' ||
  155. k === 'location' ||
  156. k === 'set-cookie'
  157. ) {
  158. const x = '--' + k
  159. resHdrNew.set(x, v)
  160. if (acehOld) {
  161. expose = expose + ',' + x
  162. }
  163. resHdrNew.delete(k)
  164. }
  165. else if (acehOld &&
  166. k !== 'cache-control' &&
  167. k !== 'content-language' &&
  168. k !== 'content-type' &&
  169. k !== 'expires' &&
  170. k !== 'last-modified' &&
  171. k !== 'pragma'
  172. ) {
  173. expose = expose + ',' + k
  174. }
  175. }
  176. if (acehOld) {
  177. expose = expose + ',--s'
  178. resHdrNew.set('--t', '1')
  179. }
  180. // verify
  181. if (rawLen) {
  182. const newLen = resHdrOld.get('content-length') || ''
  183. const badLen = (rawLen !== newLen)
  184. if (badLen) {
  185. if (retryTimes < MAX_RETRY) {
  186. urlObj = await parseYtVideoRedir(urlObj, newLen, res)
  187. if (urlObj) {
  188. return proxy(urlObj, reqInit, acehOld, rawLen, retryTimes + 1)
  189. }
  190. }
  191. return makeRes(res.body, 400, {
  192. '--error': `bad len: ${newLen}, except: ${rawLen}`,
  193. 'access-control-expose-headers': '--error',
  194. })
  195. }
  196. if (retryTimes > 1) {
  197. resHdrNew.set('--retry', retryTimes)
  198. }
  199. }
  200. let status = res.status
  201. resHdrNew.set('access-control-expose-headers', expose)
  202. resHdrNew.set('access-control-allow-origin', '*')
  203. resHdrNew.set('--s', status)
  204. resHdrNew.set('--ver', JS_VER)
  205. resHdrNew.delete('content-security-policy')
  206. resHdrNew.delete('content-security-policy-report-only')
  207. resHdrNew.delete('clear-site-data')
  208. if (status === 301 ||
  209. status === 302 ||
  210. status === 303 ||
  211. status === 307 ||
  212. status === 308
  213. ) {
  214. status = status + 10
  215. }
  216. return new Response(res.body, {
  217. status,
  218. headers: resHdrNew,
  219. })
  220. }
  221. /**
  222. * @param {URL} urlObj
  223. */
  224. function isYtUrl(urlObj) {
  225. return (
  226. urlObj.host.endsWith('.googlevideo.com') &&
  227. urlObj.pathname.startsWith('/videoplayback')
  228. )
  229. }
  230. /**
  231. * @param {URL} urlObj
  232. * @param {number} newLen
  233. * @param {Response} res
  234. */
  235. async function parseYtVideoRedir(urlObj, newLen, res) {
  236. if (newLen > 2000) {
  237. return null
  238. }
  239. if (!isYtUrl(urlObj)) {
  240. return null
  241. }
  242. try {
  243. const data = await res.text()
  244. urlObj = new URL(data)
  245. } catch (err) {
  246. return null
  247. }
  248. if (!isYtUrl(urlObj)) {
  249. return null
  250. }
  251. return urlObj
  252. }