更多示例代码

简介: 这段代码展示了EdgeRoutine的多个功能示例,包括处理不同的请求类型(如hello world、地理位置信息获取、转发请求等)、实现AB测试、多源拼接、预加载、竞速请求、简单边缘侧日志记录、重定向(基于UserAgent和地理位置信息)及拒绝爬虫访问等。每个功能通过独立函数实现,并在主处理函数中根据请求类型调用相应的处理逻辑。具体效果可参考[Yopian的示例](https://www.yopian.com/sitemap/post.xml)。

EdgeRoutine的示例源码

addEventListener("fetch", function(event) {
event.respondWith(_handleRouter(event));
});
async function _handleRouter(event) {
let json = await event.request.json();
if (json) {
let name = json.name代码效果参考:https://www.yopian.com/sitemap/post.xml
switch(name) {
case "helloworld":
return _handleHelloWorld(event);
case "geo":
return _handleGeo(event);
case "fetch":
return _handleFetch(event, json);
case "request":
return _handleRequest(event, json);
case "response":
return _handleResponse(event, json);
// Cases
case "ab-test":
return _handleABTest(event, json);
case "multi-origin":
return _handleMultipleOriginConcate(event, json);
case "prefetch":
return _handlePrefetch(event, json);
case "race":
return _handleRace(event, json);
case "esi":
return _handleESI(event, json);
case "log":
return _handleEdgeLog(event, json);
case "3xx":
return _handleRedirect3XX(event, json);
case "redirect":
return _handleRedirectGeneral(event, json);
case "deny-bot":
return _handleDenyBot(event, json);
case "waf":
return _handleWAF(event, json);
default:
break;
}
}
return new Response(
{ error : "invalid request" },
{
"status" : 403 ,
"statusText" : "Forbidden"
});
}
async function _handleHelloWorld(event) {
return new Response("Hello World!");
}
async function _handleGeo(event) {
const info = event.info;
let remote_addr = info.remote_addr;
let ip_isp_en = info.ip_isp_en;
let ip_city_en = info.ip_city_en;
let ip_region_en = info.ip_region_en;
let ip_country_en = info.ip_country_en;
let scheme = info.scheme;
let detector_device = info.detector_device;
let content = Geo: ${remote_addr}, \ ${ip_isp_en}, \ ${ip_country_en}, \ ${ip_city_en}, \ ${ip_region_en},\ ${scheme}, \ ${detector_device};
return new Response(content);
}
async function _handleFetch(event, json) {
let fetchURL = json.url;
if (fetchURL) {
return await fetch(fetchURL);
}
return fetch("http://default.ialicdn.com");
}
async function _handleRequest(event, json) {
let headers = json.headers;
let body = json.body;
const fetchInit = {
body : body,
headers: headers
};
return fetch("http://default.ialicdn.com", fetchInit);
}
async function _handleResponse(event, json) {
let resp = await fetch("http://default.ialicdn.com");
let headers = json.headers;
for (var k in headers) {
resp.headers.set(k, headers[k]);
}
return resp;
}
/* ================

  • (1) DevOps |
  • =================/
    function _shouldDoABTest(request) {
    // (1) if request's user agent match a certain string
    {
    const ua = request.headers.get("user-agent");
    if (ua && ua.match(/canary-client/)) {
    return true;
    }
    }
    // (2) whether we have special header
    {
    return request.headers.has("x-ab-test");
    }
    }
    async function _handleABTest(event, json) {
    event.request.headers.delete("content-length");
    const fetchInit = {
    method : event.request.method,
    headers: event.request.headers,
    body : "empty"
    };
    if (_shouldDoABTest(event.request)) {
    return fetch("http://default.ialicdn.com/dev", fetchInit);
    } else {
    return fetch("http://default.ialicdn.com", fetchInit);
    }
    }
    /** ==================================
  • (2) Multiple Origin Concatenation |
    ==================================*/
    async function _handleMultipleOriginConcate(event, json) {
    const respInit = {
    headers: event.request.headers,
    body : json.body
    };
    // (1) We try to concate www.alipay.com and www.tmall.com together
    let {readable, writable} = new TransformStream();
    async function controller() {
    let r1 = await fetch("http://www.alipay.com");
    let r2 = await fetch("https://www.tmall.com");
    await r1.body.pipeTo(writable, {preventClose: true});
    await r2.body.pipeTo(writable);
    }
    controller();
    return new Response(readable, respInit);
    }
    /
    ==================================*
  • (3) Precache/Prefetch |
    ==================================*/
    async function _fetchAndIgnore(url) {
    try {
    // Specify cdnProxy flag to make sure the request goes through the CDN
    let resp = await fetch(url);//, {cdnProxy: true});
    // Make sure to ignore the content otherwise the cache may not be valid
    await resp.ignore();
    } catch (e) {
    console.error("invalid URL: %s", url);
    }
    }
    async function _doPrefetchURLAsync(prefetchURL, event) {
    for (const url of prefetchURL) {
    event.waitUntil(_fetchAndIgnore(url));
    }
    }
    async function _handlePrefetch(event, json) {
    {
    const prefetchURL = json.prefetch;
    if (prefetchURL) {
    // Do not await it and let it run in background
    _doPrefetchURLAsync(prefetchURL, event);
    return new Response("Done Prefetch");
    }
    }
    return new Response("Miss Prefetch");
    }
    /
    ==================================*
  • (4) Race
    ==================================*/
    async function _handleRace(event, json) {
    let fetchList = json.fetchList;
    if (fetchList) {
    return Promise.race(fetchList.map((x) => fetch(x)));
    } else {
    return "forget to include fetchList field in your JSON";
    }
    }
    /
    ==================================*
  • (5) Simple ESI
    * ==================================/
    async function _handleESI(request, json) {
    let { readable, writable } = new TransformStream();
    let newResponse = new Response(readable);
    if (!json.esi) {
    return "forget to include template field in your JSON";
    }
    streamTransformBody(new BufferStream(json.esi), writable);
    return newResponse;
    }
    async function handleTemplate(encoder, templateKey) {
    const linkRegex = new RegExp("esi:include.src=@(.)@.*", 'gm');
    let result = linkRegex.exec(templateKey);
    let esi = "unknown";
    if (!result) {
    return encoder.encode(<${templateKey}>);
    }
    if (result[1]) {
    esi = await subRequests(result[1]);
    }
    return encoder.encode(${esi});
    }
    async function subRequests(target){
    const init = {method: 'GET'};
    let response = await fetch(target, init);
    let text = await response.text();
    return text;
    }
    async function streamTransformBody(readable, writable) {
    const startTag = "<".charCodeAt(0);
    const endTag = ">".charCodeAt(0);
    let reader = readable.getReader();
    let writer = writable.getWriter();
    let templateChunks = null;
    while (true) {
    let { done, value } = await reader.read();
    if (done) break;
    while (value.byteLength > 0) {
    if (templateChunks) {
     let end = value.indexOf(endTag);
     if (end === -1) {
       templateChunks.push(value);
       break;
     } else {
       templateChunks.push(value.subarray(0, end));
       await writer.write(await translate(templateChunks));
       templateChunks = null;
       value = value.subarray(end + 1);
     }
    
    }
    let start = value.indexOf(startTag);
    if (start === -1) {
     await writer.write(value);
     break;
    
    } else {
     await writer.write(value.subarray(0, start));
     value = value.subarray(start + 1);
     templateChunks = [];
    
    }
    }
    }
    await writer.close();
    }
    async function translate(chunks) {
    const decoder = new TextDecoder();
    let templateKey = chunks.reduce(
    (accumulator, chunk) =>
    accumulator + decoder.decode(chunk, { stream: true }), "");
    templateKey += decoder.decode();
    return handleTemplate(new TextEncoder(), templateKey);
    }
    /* ==================================
  • (6) Edge side conditional log
    * ==================================/
    async function _doEdgeLog(data, writer) {
    let resp = await fetch("http://default.ialicdn.com/log",
    {
    method : "POST",
    body : data,
    headers: [["content-type", "application/json"]]
    });
    console.log("logged");
    {
    let stream = new BufferStream("++++++++++++++++++++++++++++++\n");
    await stream.pipeTo(writer, {preventClose: true});
    }
    await resp.body.pipeTo(writer);
    }
    async function _handleEdgeLog(event, json) {
    let start= Date.now();
    let resp = await fetch("http://default.ialicdn.com", {
    method : event.request.method,
    headers: event.request.headers,
    body : json.body
    });
    // Get a promise that is fired when we send out everything
    let {readable, writable} = new TransformStream();
    // (1) first let the fetch request's response goes back and then we post
    // the log back as well internally
    let endPromise = resp.body.pipeTo(writable, {preventClose: true});
    // (2) wait for endPromise to be fired to make sure that the body has been
    // piped back to the client, and then we do the log
    event.waitUntil(endPromise.then(
    (v) => {
    let end = Date.now();
    let diff= (end - start);
    try {
     // You have to await your async promise since wait until is not
     // usable currently maybe. User can use wait until only before
     // returning the main request for now
     event.waitUntil(_doEdgeLog(`{ "cost(millisecond)" : ${diff} }`, writable));
    
    } catch (e) {
     console.error(`${e}`);
    
    }
    },
    (v) => {
    writable.abort();
    console.error("failed");
    }));
    console.error("XXXX");
    // return the response back
    return new Response(readable, {
    status: resp.status,
    headers: resp.headers
    });
    }
    /* ==================================
  • (7) redirect-3xx
    ==================================*/
    async function _handleRedirect3XX(event, json) {
    return fetch("http://www.taobao.com", {redirect: "follow"});
    }
    /
    ==================================*
  • (8) redirect
  • (1) UserAgent
  • (2) Geo information
    ==================================*/
    async function _handleRedirectGeneral(event, json) {
    const fetchInit = {
    method : event.request.method,
    body : json.body,
    headers : event.request.headers
    };
    {
    const ua = event.request.headers.get("user-agent");
    if (ua && ua.match(/firefox/i)) {
    return fetch("http://default.ialicdn.com/firefox", fetchInit);
    }
    if (ua && ua.match(/safari/i)) {
    return fetch("http://default.ialicdn.com/safari", fetchInit);
    }
    }
    {
    if (event.info.detect_device && event.info.detect_device.match(/iphone/)) {
    return fetch("http://default.ialicdn.com/iphone", fetchInit);
    }
    }
    return new Response("unknown request", {status: 403});
    }
    /
    ==================================*
  • (9) Deny bot
    ==================================*/
    async function _handleDenyBot(event, json) {
    {
    const ua = event.request.headers.get("user-agent");
    if (ua && ua.match(new RegExp("xxxspider", "i"))) {
    return new Response("Forbidden", {status: 403});
    }
    }
    return fetch("http://default.ialicdn.com");
    }
    /
    ==================================*
  • (10) Simple WAF
    * ==================================/
    async function _handleWAF(event, json) {
    let city = json.city;
    if (event.info.ip_city_en === city) {
    return new Response("Forbidden", {status: 403});
    }
    // back to origin
    return (JSON.stringify(event.info));
    }
相关文章
|
11天前
|
JSON API 开发者
示例代码是什么及其作用
示例代码是展示如何使用特定API接口的简洁代码片段,涵盖参数设置、请求发送和响应处理等步骤。它通过直观展示调用方式、减少阅读文档时间、提供可复用模板、避免常见错误,帮助开发者快速理解并应用API接口,从而降低学习成本、提高开发效率,并促进API的推广与应用。编写时应遵循简洁明了、注释清晰、涵盖常见场景及保持更新的原则,确保其易用性和准确性。
如何在文档中添加示例代码
【10月更文挑战第17天】在文档中添加示例代码是非常重要的,它可以帮助读者更好地理解和使用所介绍的内容。
|
JavaScript 前端开发
this如何使用
"this" 是 JavaScript 中的关键字,它通常用于引用当前执行上下文中的对象。
54 0
|
算法 Java
Java 方法示例代码demo(一)
Java 方法示例代码demo(一)
70 0
|
XML JSON 编解码
|
IDE 开发工具 C++
如何使用VS
如何使用VS
108 0
|
Java
Java 方法示例代码demo(二)
Java 方法示例代码demo(二)
62 0
|
Java
Java 方法示例代码demo(三)
Java 方法示例代码demo(三)
88 0
|
消息中间件 Java API
如何使用 ArrayPool
如果不停的 new 数组,可能会造成 GC 的压力,因此在 aspnetcore 中推荐使用 ArrayPool 来重用数组,本文将介绍如何使用 ArrayPool。
189 0
如何使用 ArrayPool
|
C++
如何在C中调用C++的示例代码
如何在C中调用C++的示例代码
105 0