香水电影完整版在线观看未删减|久草在线国产|欧美伊人电影|美女福利视频在线,大开色戒完整版迅雷,久艹久,动画片大尺度未删减电影

軟件開發(fā),網(wǎng)站建設(shè),微信小程序制作,抖音小程序開發(fā),湖北紅點科技有限公司

態(tài)

業(yè)

軟件開發(fā),網(wǎng)站建設(shè),微信小程序制作,抖音小程序開發(fā),湖北紅點科技有限公司
自制免費Google翻墻工具,速度很快
發(fā)布時間:2024-12-09  閱讀量:383

在 https://workers.cloudflare.com 注冊一個賬號
然后寫入下面代碼,保存并發(fā)布,然后就可以用的,速度杠杠的。
這是我測試的網(wǎng)址:https://google.ices.workers.dev/

'use strict'/**
 * static files (404.html, sw.js, conf.js)
 */const ASSET_URL = 'https://etherdream.github.io/jsproxy'const JS_VER = 10const 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-max-age': '1728000',
  }),
}/**
 * @param {any} body
 * @param {number} status
 * @param {Object<string, string>} headers
 */function makeRes(body, status = 200, headers = {}) {
  headers['--ver'] = JS_VER
  headers['access-control-allow-origin'] = '*'
  return new Response(body, {status, headers})
}/**
 * @param {string} urlStr 
 */function newUrl(urlStr) {  try {    return new URL(urlStr)
  } catch (err) {    return null
  }
}


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)  const path = urlObj.href.substr(urlObj.origin.length)  if (urlObj.protocol === 'http:') {
    urlObj.protocol = 'https:'
    return makeRes('', 301, {      'strict-transport-security': 'max-age=99999999; includeSubDomains; preload',      'location': urlObj.href,
    })
  }  if (path.startsWith('/http/')) {    return httpHandler(req, path.substr(6))
  }  switch (path) {  case '/http':    return makeRes('請更新 cfworker 到最新版本!')  case '/ws':    return makeRes('not support', 400)  case '/works':    return makeRes('it works')  default:    // static files
    return fetch(ASSET_URL + path)
  }
}/**
 * @param {Request} req
 * @param {string} pathname
 */function httpHandler(req, pathname) {  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 acehOld = false
  let rawSvr = ''
  let rawLen = ''
  let rawEtag = ''

  const reqHdrNew = new Headers(reqHdrRaw)
  reqHdrNew.set('x-jsproxy', '1')  // 此處邏輯和 http-dec-req-hdr.lua 大致相同
  // https://github.com/EtherDream/jsproxy/blob/master/lua/http-dec-req-hdr.lua
  const refer = reqHdrNew.get('referer')  const query = refer.substr(refer.indexOf('?') + 1)  if (!query) {    return makeRes('missing params', 403)
  }  const param = new URLSearchParams(query)  for (const [k, v] of Object.entries(param)) {    if (k.substr(0, 2) === '--') {      // 系統(tǒng)信息
      switch (k.substr(2)) {      case 'aceh':
        acehOld = true
        break
      case 'raw-info':
        [rawSvr, rawLen, rawEtag] = v.split('|')        break
      }
    } else {      // 還原 HTTP 請求頭
      if (v) {
        reqHdrNew.set(k, v)
      } else {
        reqHdrNew.delete(k)
      }
    }
  }  if (!param.has('referer')) {
    reqHdrNew.delete('referer')
  }  // cfworker 會把路徑中的 `//` 合并成 `/`
  const urlStr = pathname.replace(/^(https?):/+/, '$1://')  const urlObj = newUrl(urlStr)  if (!urlObj) {    return makeRes('invalid proxy url: ' + urlStr, 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 = '*'
  
  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 (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('--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
}
 最后修改:2019 年 10 月 04 日 03 : 50 PM