阅读视图

发现新文章,点击刷新页面。

Windows 11 Insider Preview 27686.1000_ZH_CN (rs_prerelease)[X64]

作者 胖氪不胖

最近微软在官方博客更新了 Windows 11 Insider Preview Build 27686 的报道(原文地址),在Canary 通道发布,大家可以在UUP上下载体验。

新版值得关注的几点变化:

1、此版本包含新的 Windows Sandbox 客户端预览版,现已通过 Microsoft Store 更新;

2、对运行 Canary 频道 Build 27686+ 的 PC 进行了优化,以延长电池寿命;

3、对“设置”>“系统”>“电源和电池”进行改进,包括在您的电脑插入电源时和使用电池时设置电源模式的功能,以及对页面进行一些其他 UI 改进;

4、在具有 HDR 显示器的 PC 上,添加了允许 HDR 视频流的选项,可在“设置”>“系统”>“显示”>“HDR”下关闭 HDR;

5、当使用format命令从命令行格式化磁盘时,将 FAT32 大小限制从 32GB 增加到 2TB。

对于新版的一些已知问题,大家可以去查看一下原博文,这里胖氪主要讲一个问题,就是有网友反馈27686.1000版本无法 “绕过硬件检测” ,针对这个问题给大家提供一个最简单的方法突破限制。

这里要用到一款工具,来自 知大Training_Win11_ISO_v1.4_by_zbezj ,下载完成后以管理员权限运行。

工具提供4个选项:

[1] 绕过Win11硬件检测

— 修改install.wim的安装属性为server,安装时自动绕过硬件检测。

[2] 绕过Win11硬件检测+无人值守

— 修改install.wim的安装属性为server,安装时自动绕过硬件检测。添加AutoUnattend.xml到ISO根目录,U盘/光盘等介质启动安装时将全自动进行安装。

[3] 绕过Win11硬件检测+跳过在线登录

— 在OOBE阶段原版ISO强制要求登录微软账户。通过此方法可以使用本地账户登录,注意需要断网。

[4] 绕过Win11硬件检测+无人值守+跳过在线登录

— 以上三种方式的合集。

说明:以上4种方式,既适用于全新安装也适用于升级安装。

使用方法:

1、将Win11原版ISO放到Source_ISO目录下。

2、以管理身份运行Training_Win11_ISO_v1.4_by_zbezj.cmd。

3、输入1~4,回车确认,等待制作完成,在批处理同目录下生成改造后的ISO镜像。

下载地址:

原版文件: 27686.1000.240809-2254.RS_PRERELEASE_CLIENTMULTI_X64FRE_ZH-CN.ISO
大小: 4958642176 字节
MD5: 93734ABC3A1C49433FE2C90F0578919E
SHA1: C05A99F7A871D9FD0C79BD8FB4A08555BC7CCC45
CRC32: 9B22B786

修复文件: 27686.1000.240809-2254.RS_PRERELEASE_CLIENTMULTI_X64FRE_ZH-CN_Trained.ISO
大小: 4958646272 字节
MD5: FB805857AD037CA8BD9B41A8AB0B3FD8
SHA1: 951EBE36435F5665A2738628E6E48D14175775BF
CRC32: 4E3D43E5

UUP下载 | 主力下载

文件: [2024.08.20]Training_Win11_ISO_Tools_v1.4_by_zbezj.7z
大小: 1694179 字节
MD5: 8D074ED5504A9F1511993403B0FCEF2D
SHA1: 6787CDDA631640AA78138E246783F2695E4445CB
CRC32: 41871C91

主力下载 | 备用下载

借助 CF 解决 Docker 镜像拉取问题

作者 Teacher Du

之前为小伙伴们提供了 Docker 镜像拉取问题的解决方案,但使用 Render 平台时出现了无法拉取部署镜像问题,Distribution Registry 又需要自行采购境外主机。本文介绍如何借助 Cloudflare 解决 Docker 镜像拉取问题。

写在前面

之前分享了如何使用 Render 平台解决 Docker 镜像拉取的问题,但 Render 平台限制了该服务部署,无法继续白嫖。

后来又分享了如何通过自行部署 Distribution Registry 解决 Docker 镜像拉取问题,但不少小伙伴留言哭穷,还是希望可以通过白嫖的方式解决该问题。

其实不少的博主都分享了如何通过 CF 的 Workers 解决镜像拉取问题,杜老师也试了一下,非常好用,就整理了一份部署教程,分享给需要的小伙伴们!

因为需通过 CF 实现,所以要准备一个可以托管到 CF 的域名。有人推荐使用 eu.org 的免费域名,但杜老师试了申请好多次都没有成功,可以点击 这里 领取一个免费的 TOP 域名「找不到领取页面可在评论区留言」

部署过程

CF 账号申请过程和域名托管步骤这里就不说了,在页面左侧找到 Workers——概述,点创建 Worker:

项目名称可自定义,也可使用 CF 自动生成的,点击右下角处部署:

待看到项目部署成功页面后,点击右上角的编辑代码:

将部署代码区域的内容调整后粘贴到页面中的代码区域,然后点击右上角的部署按钮:

返回项目页面,点击设置,切换到触发器,我们添加一个路由。这里以 docker.birdteam.net 为例,区域则选择顶级域,最后点击右下角处添加路由:

切换到域名 DNS 记录页面,添加一个 A 类记录,主机名填写 docker,记录值可随意填写。注意务必开启代理状态,保存即可:

部署代码

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
let hub_host = 'registry-1.docker.io'
const auth_url = 'https://auth.docker.io'
let workers_url = 'https://docker.birdteam.net'
let UA = ['netcraft'];
function routeByHosts(host) {
const routes = {
"quay": "quay.io",
"gcr": "gcr.io",
"k8s-gcr": "k8s.gcr.io",
"k8s": "registry.k8s.io",
"ghcr": "ghcr.io",
"cloudsmith": "docker.cloudsmith.io",
"test": "registry-1.docker.io",
};
if (host in routes) return [ routes[host], false ];
else return [ hub_host, true ];
}
const PREFLIGHT_INIT = {
headers: new Headers({
'access-control-allow-origin': '*',
'access-control-allow-methods': 'GET,POST,PUT,PATCH,TRACE,DELETE,HEAD,OPTIONS',
'access-control-max-age': '1728000',
}),
}
function makeRes(body, status = 200, headers = {}) {
headers['access-control-allow-origin'] = '*'
return new Response(body, { status, headers })
}
function newUrl(urlStr) {
try {
return new URL(urlStr)
} catch (err) {
return null
}
}
function isUUID(uuid) {
const uuidRegex = /^[0-9a-f]{8}-[0-9a-f]{4}-[4][0-9a-f]{3}-[89ab][0-9a-f]{3}-[0-9a-f]{12}$/i;
return uuidRegex.test(uuid);
}
async function nginx() {
const text = `
<!DOCTYPE html>
<html>
<head>
<title>Welcome to nginx!</title>
<style>
body {
width: 35em;
margin: 0 auto;
font-family: Tahoma, Verdana, Arial, sans-serif;
}
</style>
</head>
<body>
<h1>Welcome to nginx!</h1>
<p>If you see this page, the nginx web server is successfully installed and
working. Further configuration is required.</p>
<p>For online documentation and support please refer to
<a href="http://nginx.org/">nginx.org</a>.<br/>
Commercial support is available at
<a href="http://nginx.com/">nginx.com</a>.</p>
<p><em>Thank you for using nginx.</em></p>
</body>
</html>
`
return text ;
}
export default {
async fetch(request, env, ctx) {
const getReqHeader = (key) => request.headers.get(key);
let url = new URL(request.url);
const userAgentHeader = request.headers.get('User-Agent');
const userAgent = userAgentHeader ? userAgentHeader.toLowerCase() : "null";
if (env.UA) UA = UA.concat(await ADD(env.UA));
workers_url = `https://${url.hostname}`;
const pathname = url.pathname;
const hostname = url.searchParams.get('hubhost') || url.hostname;
const hostTop = hostname.split('.')[0];
const checkHost = routeByHosts(hostTop);
hub_host = checkHost[0];
const fakePage = checkHost[1];
console.log(`域名头部: ${hostTop}\n反代地址: ${hub_host}\n伪装首页: ${fakePage}`);
const isUuid = isUUID(pathname.split('/')[1].split('/')[0]);
if (UA.some(fxxk => userAgent.includes(fxxk)) && UA.length > 0){
return new Response(await nginx(), {
headers: {
'Content-Type': 'text/html; charset=UTF-8',
},
});
}
const conditions = [
isUuid,
pathname.includes('/_'),
pathname.includes('/r'),
pathname.includes('/v2/user'),
pathname.includes('/v2/orgs'),
pathname.includes('/v2/_catalog'),
pathname.includes('/v2/categories'),
pathname.includes('/v2/feature-flags'),
pathname.includes('search'),
pathname.includes('source'),
pathname === '/',
pathname === '/favicon.ico',
pathname === '/auth/profile',
];
if (conditions.some(condition => condition) && (fakePage === true || hostTop == 'docker')) {
if (env.URL302){
return Response.redirect(env.URL302, 302);
} else if (env.URL){
if (env.URL.toLowerCase() == 'nginx'){
return new Response(await nginx(), {
headers: {
'Content-Type': 'text/html; charset=UTF-8',
},
});
} else return fetch(new Request(env.URL, request));
}
const newUrl = new URL("https://registry.hub.docker.com" + pathname + url.search);
const headers = new Headers(request.headers);
headers.set('Host', 'registry.hub.docker.com');
const newRequest = new Request(newUrl, {
method: request.method,
headers: headers,
body: request.method !== 'GET' && request.method !== 'HEAD' ? await request.blob() : null,
redirect: 'follow'
});
return fetch(newRequest);
}
if (!/%2F/.test(url.search) && /%3A/.test(url.toString())) {
let modifiedUrl = url.toString().replace(/%3A(?=.*?&)/, '%3Alibrary%2F');
url = new URL(modifiedUrl);
console.log(`handle_url: ${url}`)
}
if (url.pathname.includes('/token')) {
let token_parameter = {
headers: {
'Host': 'auth.docker.io',
'User-Agent': getReqHeader("User-Agent"),
'Accept': getReqHeader("Accept"),
'Accept-Language': getReqHeader("Accept-Language"),
'Accept-Encoding': getReqHeader("Accept-Encoding"),
'Connection': 'keep-alive',
'Cache-Control': 'max-age=0'
}
};
let token_url = auth_url + url.pathname + url.search
return fetch(new Request(token_url, request), token_parameter)
}
if (/^\/v2\/[^/]+\/[^/]+\/[^/]+$/.test(url.pathname) && !/^\/v2\/library/.test(url.pathname)) {
url.pathname = url.pathname.replace(/\/v2\//, '/v2/library/');
console.log(`modified_url: ${url.pathname}`)
}
url.hostname = hub_host;
let parameter = {
headers: {
'Host': hub_host,
'User-Agent': getReqHeader("User-Agent"),
'Accept': getReqHeader("Accept"),
'Accept-Language': getReqHeader("Accept-Language"),
'Accept-Encoding': getReqHeader("Accept-Encoding"),
'Connection': 'keep-alive',
'Cache-Control': 'max-age=0'
},
cacheTtl: 3600
};
if (request.headers.has("Authorization")) {
parameter.headers.Authorization = getReqHeader("Authorization");
}
let original_response = await fetch(new Request(url, request), parameter)
let original_response_clone = original_response.clone();
let original_text = original_response_clone.body;
let response_headers = original_response.headers;
let new_response_headers = new Headers(response_headers);
let status = original_response.status;
if (new_response_headers.get("Www-Authenticate")) {
let auth = new_response_headers.get("Www-Authenticate");
let re = new RegExp(auth_url, 'g');
new_response_headers.set("Www-Authenticate", response_headers.get("Www-Authenticate").replace(re, workers_url));
}
if (new_response_headers.get("Location")) {
return httpHandler(request, new_response_headers.get("Location"))
}
let response = new Response(original_text, {
status,
headers: new_response_headers
})
return response;
}
};
function httpHandler(req, pathname) {
const reqHdrRaw = req.headers
if (req.method === 'OPTIONS' &&
reqHdrRaw.has('access-control-request-headers')
) {
return new Response(null, PREFLIGHT_INIT)
}
let rawLen = ''
const reqHdrNew = new Headers(reqHdrRaw)
const refer = reqHdrNew.get('referer')
let urlStr = pathname
const urlObj = newUrl(urlStr)
const reqInit = {
method: req.method,
headers: reqHdrNew,
redirect: 'follow',
body: req.body
}
return proxy(urlObj, reqInit, rawLen)
}
async function proxy(urlObj, reqInit, rawLen) {
const res = await fetch(urlObj.href, reqInit)
const resHdrOld = res.headers
const resHdrNew = new Headers(resHdrOld)
if (rawLen) {
const newLen = resHdrOld.get('content-length') || ''
const badLen = (rawLen !== newLen)
if (badLen) {
return makeRes(res.body, 400, {
'--error': `bad len: ${newLen}, except: ${rawLen}`,
'access-control-expose-headers': '--error',
})
}
}
const status = res.status
resHdrNew.set('access-control-expose-headers', '*')
resHdrNew.set('access-control-allow-origin', '*')
resHdrNew.set('Cache-Control', 'max-age=1500')
resHdrNew.delete('content-security-policy')
resHdrNew.delete('content-security-policy-report-only')
resHdrNew.delete('clear-site-data')
return new Response(res.body, {
status,
headers: resHdrNew
})
}
async function ADD(envadd) {
var addtext = envadd.replace(/[ |"'\r\n]+/g, ',').replace(/,+/g, ',');
//console.log(addtext);
if (addtext.charAt(0) == ',') addtext = addtext.slice(1);
if (addtext.charAt(addtext.length -1) == ',') addtext = addtext.slice(0, addtext.length - 1);
const add = addtext.split(',');
//console.log(add);
return add ;
}

注意:在粘贴前,需要调整第三行的代码,将 docker.birdteam.net 改为您的域名。

自建 Docker 镜像存储库解决拉取问题

作者 Teacher Du

Distribution Registry 是一个开源镜像仓库,用于存储和管理 Docker 镜像。它允许您在 Linux 服务器上创建私有的 Docker 镜像仓库,以便团队成员共享、访问镜像,也可用于加速拉取「解决境内拉取问题」

主要功能

镜像存储、管理。Distribution Registry 提供功能强大的仓库系统,用于存储和管理 Docker 镜像,方便团队成员之间的共享和访问。

可私有化部署。您可以在自己的 Linux 服务器上搭建私有的 Distribution Registry,以满足安全和隐私要求。

访问控制。支持设置访问权限,可以控制谁可以拉取和推送镜像,以保护您的镜像和数据的安全性。

标签、版本管理。可以为镜像设置标签和版本,方便对镜像进行分类和管理。

兼容性好。Distribution Registry 兼容 Docker 镜像仓库的标准 API,可以使用 Docker CLI 或其它 Docker 客户端工具与之交互。

支持指定上游源 URL。可通过指定上游源 URL 以加速镜像拉取,解决境内拉取问题。

安装配置

这里推荐使用 Docker 来部署,将下面的内容保存到 docker-compose.yml

1
2
3
4
5
6
7
8
services:
docker-registry:
image: registry:2.8.3
restart: always
ports:
- 5000:5000
volumes:
- ./data:/var/lib/registry

然后使用下面命令启动即可:

1
docker-compose up -d

当出现上游源 URL 无法使用时,可以通过添加下面参数来指定上游源地址:

1
2
3
4
5
proxy:
remoteurl: https://registry-1.docker.io
username: [username]
password: [password]
ttl: 168h

可以通过参数来指定缓存的方式「注意两个 cache 选其一,互相冲突」

1
2
3
4
5
6
cache:
blobdescriptor: redis
blobdescriptorsize: 10000
cache:
blobdescriptor: inmemory
blobdescriptorsize: 10000

参数说明

上文中出现的参数说明如下:

参数必填描述
remoteurlYDocker Hub 上存储库 URL。
usernameN私有存储库中注册的用户名。
passwordN私有存储库中注册密码。
ttlN代理缓存过期时间,0 为禁止缓存过期。
blobdescriptorY指定缓存方式,inmemory 为内存缓存,redis 则为 Redis 缓存。
blobdescriptorsizeN要存储在缓存中的描述符数限制。如果参数设置为 0,则允许缓存在没有大小限制的情况下增长。

注意事项

修改客户端的配置文件,默认路径为/etc/docker/daemon.json,添加如下内容:

1
2
3
4
5
{
"registry-mirrors": [
"http://IP:5000"
]
}

因为默认不支持 HTTPS,需使用 Nginx 配置反向代理。运行下面的命令重启 Docker 服务:

1
systemctl restart docker
❌