index.js 6.1 KB

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