| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284 | 
							- 'use strict'
 
- /**
 
-  * static files (404.html, sw.js, conf.js)
 
-  */
 
- const ASSET_URL = 'https://zjcqoo.github.io'
 
- const JS_VER = 10
 
- const MAX_RETRY = 1
 
- /** @type {RequestInit} */
 
- const PREFLIGHT_INIT = {
 
-   status: 204,
 
-   headers: new Headers({
 
-     'access-control-allow-origin': '*',
 
-     'access-control-allow-methods': 'GET,POST,PUT,PATCH,TRACE,DELETE,HEAD,OPTIONS',
 
-     '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',
 
-     'access-control-max-age': '1728000',
 
-   }),
 
- }
 
- /**
 
-  * @param {any} body
 
-  * @param {number} status
 
-  * @param {Object<string, string>} headers
 
-  */
 
- function makeRes(body, status = 200, headers = {}) {
 
-   headers['cache-control'] = 'no-cache'
 
-   headers['vary'] = '--url'
 
-   headers['--ver'] = JS_VER
 
-   headers['access-control-allow-origin'] = '*'
 
-   return new Response(body, {status, headers})
 
- }
 
- addEventListener('fetch', e => {
 
-   const ret = fetchHandler(e)
 
-     .catch(err => makeRes('cfworker error:\n' + err.stack, 502))
 
-   e.respondWith(ret)
 
- })
 
- /**
 
-  * @param {FetchEvent} e 
 
-  */
 
- async function fetchHandler(e) {
 
-   const req = e.request
 
-   const urlStr = req.url
 
-   const urlObj = new URL(urlStr)
 
-   if (urlObj.protocol === 'http:') {
 
-     urlObj.protocol = 'https:'
 
-     return makeRes('', 301, {
 
-       'strict-transport-security': 'max-age=99999999; includeSubDomains; preload',
 
-       'location': urlObj.href,
 
-     })
 
-   }
 
-   switch (urlObj.pathname) {
 
-   case '/http':
 
-     return httpHandler(req)
 
-   case '/ws':
 
-     return makeRes('not support', 400)
 
-   case '/works':
 
-     return makeRes('it works')
 
-   default:
 
-     // static files
 
-     return fetch(ASSET_URL + urlObj.pathname)
 
-   }
 
- }
 
- /**
 
-  * @param {Request} req
 
-  */
 
- function httpHandler(req) {
 
-   const reqHdrRaw = req.headers
 
-   if (reqHdrRaw.has('x-jsproxy')) {
 
-     return Response.error()
 
-   }
 
-   // preflight
 
-   if (req.method === 'OPTIONS' &&
 
-       reqHdrRaw.has('access-control-request-headers')
 
-   ) {
 
-     return new Response(null, PREFLIGHT_INIT)
 
-   }
 
-   let urlObj = null
 
-   let extHdrs = null
 
-   let acehOld = false
 
-   let rawSvr = ''
 
-   let rawLen = ''
 
-   let rawEtag = ''
 
-   const reqHdrNew = new Headers(reqHdrRaw)
 
-   reqHdrNew.set('x-jsproxy', '1')
 
-   for (const [k, v] of reqHdrRaw.entries()) {
 
-     if (!k.startsWith('--')) {
 
-       continue
 
-     }
 
-     reqHdrNew.delete(k)
 
-     const k2 = k.substr(2)
 
-     switch (k2) {
 
-     case 'url':
 
-       urlObj = new URL(v)
 
-       break
 
-     case 'aceh':
 
-       acehOld = true
 
-       break
 
-     case 'raw-info':
 
-       [rawSvr, rawLen, rawEtag] = v.split('|')
 
-       break
 
-     case 'level':
 
-     case 'mode':
 
-     case 'type':
 
-       break
 
-     case 'ext':
 
-       extHdrs = JSON.parse(v)
 
-       break
 
-     default:
 
-       if (v) {
 
-         reqHdrNew.set(k2, v)
 
-       } else {
 
-         reqHdrNew.delete(k2)
 
-       }
 
-       break
 
-     }
 
-   }
 
-   if (extHdrs) {
 
-     for (const [k, v] of Object.entries(extHdrs)) {
 
-       reqHdrNew.set(k, v)
 
-     }
 
-   }
 
-   if (!urlObj) {
 
-     return makeRes('missing url param', 403)
 
-   }
 
-   /** @type {RequestInit} */
 
-   const reqInit = {
 
-     method: req.method,
 
-     headers: reqHdrNew,
 
-     redirect: 'manual',
 
-   }
 
-   if (req.method === 'POST') {
 
-     reqInit.body = req.body
 
-   }
 
-   return proxy(urlObj, reqInit, acehOld, rawLen, 0)
 
- }
 
- /**
 
-  * 
 
-  * @param {URL} urlObj 
 
-  * @param {RequestInit} reqInit 
 
-  * @param {number} retryTimes 
 
-  */
 
- async function proxy(urlObj, reqInit, acehOld, rawLen, retryTimes) {
 
-   const res = await fetch(urlObj.href, reqInit)
 
-   const resHdrOld = res.headers
 
-   const resHdrNew = new Headers(resHdrOld)
 
-   let expose = '*'
 
-   let vary = '--url'
 
-   
 
-   for (const [k, v] of resHdrOld.entries()) {
 
-     if (k === 'access-control-allow-origin' ||
 
-         k === 'access-control-expose-headers' ||
 
-         k === 'location' ||
 
-         k === 'set-cookie'
 
-     ) {
 
-       const x = '--' + k
 
-       resHdrNew.set(x, v)
 
-       if (acehOld) {
 
-         expose = expose + ',' + x
 
-       }
 
-       resHdrNew.delete(k)
 
-     }
 
-     else if (k === 'vary') {
 
-       vary = vary + ',' + v
 
-     }
 
-     else if (acehOld &&
 
-       k !== 'cache-control' &&
 
-       k !== 'content-language' &&
 
-       k !== 'content-type' &&
 
-       k !== 'expires' &&
 
-       k !== 'last-modified' &&
 
-       k !== 'pragma'
 
-     ) {
 
-       expose = expose + ',' + k
 
-     }
 
-   }
 
-   if (acehOld) {
 
-     expose = expose + ',--s'
 
-     resHdrNew.set('--t', '1')
 
-   }
 
-   // verify
 
-   if (rawLen) {
 
-     const newLen = resHdrOld.get('content-length') || ''
 
-     const badLen = (rawLen !== newLen)
 
-     if (badLen) {
 
-       if (retryTimes < MAX_RETRY) {
 
-         urlObj = await parseYtVideoRedir(urlObj, newLen, res)
 
-         if (urlObj) {
 
-           return proxy(urlObj, reqInit, acehOld, rawLen, retryTimes + 1)
 
-         }
 
-       }
 
-       return makeRes(res.body, 400, {
 
-         '--error': `bad len: ${newLen}, except: ${rawLen}`,
 
-         'access-control-expose-headers': '--error',
 
-       })
 
-     }
 
-     if (retryTimes > 1) {
 
-       resHdrNew.set('--retry', retryTimes)
 
-     }
 
-   }
 
-   let status = res.status
 
-   resHdrNew.set('access-control-expose-headers', expose)
 
-   resHdrNew.set('access-control-allow-origin', '*')
 
-   resHdrNew.set('vary', vary)
 
-   resHdrNew.set('--s', status)
 
-   resHdrNew.set('--ver', JS_VER)
 
-   resHdrNew.delete('content-security-policy')
 
-   resHdrNew.delete('content-security-policy-report-only')
 
-   resHdrNew.delete('clear-site-data')
 
-   if (status === 301 ||
 
-       status === 302 ||
 
-       status === 303 ||
 
-       status === 307 ||
 
-       status === 308
 
-   ) {
 
-     status = status + 10
 
-   }
 
-   return new Response(res.body, {
 
-     status,
 
-     headers: resHdrNew,
 
-   })
 
- }
 
- /**
 
-  * @param {URL} urlObj 
 
-  */
 
- function isYtUrl(urlObj) {
 
-   return (
 
-     urlObj.host.endsWith('.googlevideo.com') &&
 
-     urlObj.pathname.startsWith('/videoplayback')
 
-   )
 
- }
 
- /**
 
-  * @param {URL} urlObj 
 
-  * @param {number} newLen 
 
-  * @param {Response} res 
 
-  */
 
- async function parseYtVideoRedir(urlObj, newLen, res) {
 
-   if (newLen > 2000) {
 
-     return null
 
-   }
 
-   if (!isYtUrl(urlObj)) {
 
-     return null
 
-   }
 
-   try {
 
-     const data = await res.text()
 
-     urlObj = new URL(data)
 
-   } catch (err) {
 
-     return null
 
-   }
 
-   if (!isYtUrl(urlObj)) {
 
-     return null
 
-   }
 
-   return urlObj
 
- }
 
 
  |