index.html
                        
                             · 3.3 KiB · HTML
                        
                    
                    
                      
                        Исходник
                      
                    
                      
                    
                        
                          
                        
                    
                    
                
                
                
            <!DOCTYPE html>
<html lang="zh-CN">
<head>
    <meta charset="utf-8">
    <meta name="viewport" content="width=device-width, initial-scale=1">
    <title>V2镜像使用说明</title>
    <style>
        body {
            font-family: 'Roboto', sans-serif;
            margin: 0;
            padding: 0;
            background: linear-gradient(135deg, #e0f7fa, #e1bee7);
            color: #333;
        }
        .header {
            background: linear-gradient(135deg, #667eea, #764ba2);
            color: #fff;
            padding: 20px 0;
            text-align: center;
            box-shadow: 0 2px 4px rgba(0, 0, 0, 0.1);
            animation: fadeIn 1.5s;
        }
        .container {
            max-width: 800px;
            margin: 40px auto;
            padding: 20px;
            background-color: #fff;
            box-shadow: 0 4px 8px rgba(0, 0, 0, 0.1);
            border-radius: 10px;
            animation: slideInUp 1s;
        }
        .content {
            margin-bottom: 20px;
        }
        .footer {
            text-align: center;
            padding: 20px 0;
            background-color: #333;
            color: #fff;
            animation: fadeIn 1.5s;
        }
        pre {
            background-color: #272822;
            color: #f8f8f2;
            padding: 15px;
            border-radius: 5px;
            overflow-x: auto;
        }
        code {
            font-family: 'Source Code Pro', monospace;
        }
        a {
            color: #4CAF50;
            text-decoration: none;
            transition: color 0.3s;
        }
        a:hover {
            color: #388E3C;
            text-decoration: underline;
        }
        @media (max-width: 600px) {
            .container {
                margin: 20px;
                padding: 15px;
            }
            .header {
                padding: 15px 0;
            }
        }
        @keyframes fadeIn {
            from { opacity: 0; }
            to { opacity: 1; }
        }
        @keyframes slideInUp {
            from { transform: translateY(20px); opacity: 0; }
            to { transform: translateY(0); opacity: 1; }
        }
    </style>
    <link href="https://fonts.googleapis.com/css2?family=Roboto:wght@400;700&family=Source+Code+Pro:wght@400;700&display=swap" rel="stylesheet">
</head>
<body>
    <div class="header">
        <h1>ghcr镜像使用说明</h1>
    </div>
    <div class="container">
        <div class="content">
            <p>为了加速镜像拉取,你可以使用以下命令设置 registry mirror:</p>
            <pre><code>sudo tee /etc/docker/daemon.json <<EOF
{
    "registry-mirrors": ["https://{{host}}"]
}
EOF</code></pre>
            <p>为了避免 Worker 用量耗尽,你可以手动 pull 镜像然后 re-tag 之后 push 至本地镜像仓库:</p>
            <pre><code>docker pull {{host}}/library/alpine:latest # 拉取 library 镜像
docker pull {{host}}/coredns/coredns:latest # 拉取 coredns 镜像</code></pre>
        </div>
    </div>
    <div class="footer">
        <p>Powered by Cloudflare Workers</p>
        <p>免责声明:本服务及其内容按“现状”提供,不提供任何形式的明示或暗示的保证,包括但不限于对适销性、适用性的任何保证。使用本服务存在的风险由用户自行承担。</p>
    </div>
</body>
</html>
                | 1 | <!DOCTYPE html> | 
| 2 | <html lang="zh-CN"> | 
| 3 | <head> | 
| 4 | <meta charset="utf-8"> | 
| 5 | <meta name="viewport" content="width=device-width, initial-scale=1"> | 
| 6 | <title>V2镜像使用说明</title> | 
| 7 | <style> | 
| 8 | body { | 
| 9 | font-family: 'Roboto', sans-serif; | 
| 10 | margin: 0; | 
| 11 | padding: 0; | 
| 12 | background: linear-gradient(135deg, #e0f7fa, #e1bee7); | 
| 13 | color: #333; | 
| 14 | } | 
| 15 | .header { | 
| 16 | background: linear-gradient(135deg, #667eea, #764ba2); | 
| 17 | color: #fff; | 
| 18 | padding: 20px 0; | 
| 19 | text-align: center; | 
| 20 | box-shadow: 0 2px 4px rgba(0, 0, 0, 0.1); | 
| 21 | animation: fadeIn 1.5s; | 
| 22 | } | 
| 23 | .container { | 
| 24 | max-width: 800px; | 
| 25 | margin: 40px auto; | 
| 26 | padding: 20px; | 
| 27 | background-color: #fff; | 
| 28 | box-shadow: 0 4px 8px rgba(0, 0, 0, 0.1); | 
| 29 | border-radius: 10px; | 
| 30 | animation: slideInUp 1s; | 
| 31 | } | 
| 32 | .content { | 
| 33 | margin-bottom: 20px; | 
| 34 | } | 
| 35 | .footer { | 
| 36 | text-align: center; | 
| 37 | padding: 20px 0; | 
| 38 | background-color: #333; | 
| 39 | color: #fff; | 
| 40 | animation: fadeIn 1.5s; | 
| 41 | } | 
| 42 | pre { | 
| 43 | background-color: #272822; | 
| 44 | color: #f8f8f2; | 
| 45 | padding: 15px; | 
| 46 | border-radius: 5px; | 
| 47 | overflow-x: auto; | 
| 48 | } | 
| 49 | code { | 
| 50 | font-family: 'Source Code Pro', monospace; | 
| 51 | } | 
| 52 | a { | 
| 53 | color: #4CAF50; | 
| 54 | text-decoration: none; | 
| 55 | transition: color 0.3s; | 
| 56 | } | 
| 57 | a:hover { | 
| 58 | color: #388E3C; | 
| 59 | text-decoration: underline; | 
| 60 | } | 
| 61 | @media (max-width: 600px) { | 
| 62 | .container { | 
| 63 | margin: 20px; | 
| 64 | padding: 15px; | 
| 65 | } | 
| 66 | .header { | 
| 67 | padding: 15px 0; | 
| 68 | } | 
| 69 | } | 
| 70 | @keyframes fadeIn { | 
| 71 | from { opacity: 0; } | 
| 72 | to { opacity: 1; } | 
| 73 | } | 
| 74 | @keyframes slideInUp { | 
| 75 | from { transform: translateY(20px); opacity: 0; } | 
| 76 | to { transform: translateY(0); opacity: 1; } | 
| 77 | } | 
| 78 | </style> | 
| 79 | <link href="https://fonts.googleapis.com/css2?family=Roboto:wght@400;700&family=Source+Code+Pro:wght@400;700&display=swap" rel="stylesheet"> | 
| 80 | </head> | 
| 81 | <body> | 
| 82 | <div class="header"> | 
| 83 | <h1>ghcr镜像使用说明</h1> | 
| 84 | </div> | 
| 85 | <div class="container"> | 
| 86 | <div class="content"> | 
| 87 | <p>为了加速镜像拉取,你可以使用以下命令设置 registry mirror:</p> | 
| 88 | <pre><code>sudo tee /etc/docker/daemon.json <<EOF | 
| 89 | { | 
| 90 | "registry-mirrors": ["https://{{host}}"] | 
| 91 | } | 
| 92 | EOF</code></pre> | 
| 93 | <p>为了避免 Worker 用量耗尽,你可以手动 pull 镜像然后 re-tag 之后 push 至本地镜像仓库:</p> | 
| 94 | <pre><code>docker pull {{host}}/library/alpine:latest # 拉取 library 镜像 | 
| 95 | docker pull {{host}}/coredns/coredns:latest # 拉取 coredns 镜像</code></pre> | 
| 96 | </div> | 
| 97 | </div> | 
| 98 | <div class="footer"> | 
| 99 | <p>Powered by Cloudflare Workers</p> | 
| 100 | <p>免责声明:本服务及其内容按“现状”提供,不提供任何形式的明示或暗示的保证,包括但不限于对适销性、适用性的任何保证。使用本服务存在的风险由用户自行承担。</p> | 
| 101 | </div> | 
| 102 | </body> | 
| 103 | </html> | 
| 104 | 
                    
                        
                        worker.js
                        
                             · 1.4 KiB · JavaScript
                        
                    
                    
                      
                        Исходник
                      
                    
                      
                    
                        
                          
                        
                    
                    
                
                
                
            import HTML from './index.html';
export default {
    async fetch(request) {
        const url = new URL(request.url);
        const path = url.pathname;
        const originalHost = request.headers.get("host");
        const registryHost = "ghcr.io";
        if (path.startsWith("/v2/")) {
            const headers = new Headers(request.headers);
            headers.set("host", registryHost);
            const registryUrl = `https://${registryHost}${path}`;
            const registryRequest = new Request(registryUrl, {
                method: request.method,
                headers: headers,
                body: request.body,
                redirect: "follow",
            });
            const registryResponse = await fetch(registryRequest);
            const responseHeaders = new Headers(registryResponse.headers);
            responseHeaders.set("access-control-allow-origin", originalHost);
            responseHeaders.set("access-control-allow-headers", "Authorization");
            return new Response(registryResponse.body, {
                status: registryResponse.status,
                statusText: registryResponse.statusText,
                headers: responseHeaders,
            });
        } else {
            return new Response(HTML.replace(/{{host}}/g, originalHost), {
                status: 200,
                headers: {
                    "content-type": "text/html"
                }
            });
        }
    }
}
                | 1 | import HTML from './index.html'; | 
| 2 | |
| 3 | export default { | 
| 4 | async fetch(request) { | 
| 5 | const url = new URL(request.url); | 
| 6 | const path = url.pathname; | 
| 7 | const originalHost = request.headers.get("host"); | 
| 8 | const registryHost = "ghcr.io"; | 
| 9 | |
| 10 | if (path.startsWith("/v2/")) { | 
| 11 | const headers = new Headers(request.headers); | 
| 12 | headers.set("host", registryHost); | 
| 13 | |
| 14 | const registryUrl = `https://${registryHost}${path}`; | 
| 15 | const registryRequest = new Request(registryUrl, { | 
| 16 | method: request.method, | 
| 17 | headers: headers, | 
| 18 | body: request.body, | 
| 19 | redirect: "follow", | 
| 20 | }); | 
| 21 | |
| 22 | const registryResponse = await fetch(registryRequest); | 
| 23 | const responseHeaders = new Headers(registryResponse.headers); | 
| 24 | responseHeaders.set("access-control-allow-origin", originalHost); | 
| 25 | responseHeaders.set("access-control-allow-headers", "Authorization"); | 
| 26 | |
| 27 | return new Response(registryResponse.body, { | 
| 28 | status: registryResponse.status, | 
| 29 | statusText: registryResponse.statusText, | 
| 30 | headers: responseHeaders, | 
| 31 | }); | 
| 32 | } else { | 
| 33 | return new Response(HTML.replace(/{{host}}/g, originalHost), { | 
| 34 | status: 200, | 
| 35 | headers: { | 
| 36 | "content-type": "text/html" | 
| 37 | } | 
| 38 | }); | 
| 39 | } | 
| 40 | } | 
| 41 | } | 
| 42 |