diff options
author | 2023-12-15 21:25:00 +0800 | |
---|---|---|
committer | 2024-01-03 22:58:15 +0800 | |
commit | 4b70534bb4fce2ab978d0628660f543a8076ca93 (patch) | |
tree | 271dc54d1b7b51643095640fe69c659ba7693973 | |
parent | f401b0ce48e19f1e25af14b3a78b4d9ef8998e73 (diff) | |
download | myweb-4b70534bb4fce2ab978d0628660f543a8076ca93.tar.gz myweb-4b70534bb4fce2ab978d0628660f543a8076ca93.zip |
Add git server and python http server configuration
- how to build a git bare repo server using my server
- add sth to python http.server to show files in server
- in order to do this, split the aliyun_cloud.md into several parts
-rw-r--r-- | .gitignore | 3 | ||||
-rw-r--r-- | code/highlight.html | 350 | ||||
-rw-r--r-- | code/linux/aliyun_cloud.html | 568 | ||||
-rw-r--r-- | code/linux/aliyun_cloud.md | 628 | ||||
-rw-r--r-- | code/linux/gitserver.html | 364 | ||||
-rw-r--r-- | code/linux/gitserver.md | 368 | ||||
-rw-r--r-- | code/linux/httpmyserver.html | 277 | ||||
-rw-r--r-- | code/linux/httpmyserver.md | 291 | ||||
-rw-r--r-- | code/linux/vim.html | 833 | ||||
-rw-r--r-- | code/linux/vim.md | 413 | ||||
-rw-r--r-- | code/stdc++.html | 892 | ||||
-rw-r--r-- | common/CSS/MySet.css | 3 | ||||
-rw-r--r-- | common/CSS/huang.css | 2 | ||||
-rw-r--r-- | common/script4works.html | 28 | ||||
-rw-r--r-- | common/script4works.js | 28 | ||||
-rwxr-xr-x | md2html.sh | 28 | ||||
-rw-r--r-- | works/education.html | 2 | ||||
-rw-r--r-- | works/poetry.html | 87 |
18 files changed, 2281 insertions, 2884 deletions
@@ -1,6 +1,7 @@ | |||
1 | # To ignore all files used to push or run | 1 | # To ignore all files used to push or run |
2 | *.bat | 2 | *.bat |
3 | *.sh | 3 | push.sh |
4 | *.exe | 4 | *.exe |
5 | .*.sw* | ||
5 | /.vscode | 6 | /.vscode |
6 | */html2js | 7 | */html2js |
diff --git a/code/highlight.html b/code/highlight.html index a0d9b61..840a3fb 100644 --- a/code/highlight.html +++ b/code/highlight.html | |||
@@ -6,76 +6,48 @@ | |||
6 | <title>我的代码块</title> | 6 | <title>我的代码块</title> |
7 | <link rel="stylesheet" href="https://www.qin-juan-ge-zhu.top/common/CSS/pandoc.css"> | 7 | <link rel="stylesheet" href="https://www.qin-juan-ge-zhu.top/common/CSS/pandoc.css"> |
8 | <script type="text/javascript" src="https://www.qin-juan-ge-zhu.top/common/script4code.js"></script> | 8 | <script type="text/javascript" src="https://www.qin-juan-ge-zhu.top/common/script4code.js"></script> |
9 | <script src="https://www.qin-juan-ge-zhu.top/common/js/myhighlight.js"></script> | ||
10 | </head> | 9 | </head> |
11 | 10 | ||
12 | <body> | 11 | <body> |
13 | <p class="title">博客代码高亮与一键复制</p> | ||
14 | <div class="pandoc"> | 12 | <div class="pandoc"> |
15 | <span class="menu"> | 13 | <div class="main"> |
14 | <p class="title">博客代码高亮与一键复制</p> | ||
15 | <p>作为一名计算机学生,代码是生活里必不可少的组件。因而自年初开始学习前端、购买域名建站的时候,我就曾想过如何在网页里实现代码高亮。奈何才疏学浅,就此搁置。</p> | ||
16 | <p>上个月在搜索如何用代码自行实现 git 基本功能的时候,误打误撞进了大佬 Nikita 的<a href="https://www.leshenko.net/p/ugit/#">μgit | ||
17 | 教程</a>,惊为天人,不仅内容详细,排版也极为美观,尤其是代码的 git diff 显示,可以清晰地看到每一步做了哪些改变,仿佛大佬亲自莅临指导。于是在开启 μgit | ||
18 | 学习(<del>尚未有所推进,笑死</del>,算是个坑,<strong>以后补吧</strong>)的同时,与黄佬共同学习该教程所用到的代码高亮办法,发现是一个 js 插件 | ||
19 | highlight.min.js,探索过程中又找到了一个在此基础上以表格办法为代码添加行号的 js 插件 highlightjs-line-numbers.js,形成了一份草创的为 git 仓库提供 diff | ||
20 | 显示的<a | ||
21 | href="https://www.qin-juan-ge-zhu.top/code/others'code/highlight.html">网页</a>,按照目前的理解尚需一些其他工具,因此该网页几乎等同于完全不会,只能说“欲知后事如何,且听下回分解”了。(一段话两个坑?挖坑小能手实锤!) | ||
22 | </p> | ||
23 | <p>同时,一个多月来我为服务器 vim 做的配置 vimrc 变化较大。万一身边有人学呢?或者,年底服务期到期了这些设置总需要有个备份吧?故而近来想着重新写一份最新的。于是我碰到了之前无法解决的问题:之前 | ||
24 | Markdown | ||
25 | 导出为 html 所使用的工具是 typora,在导出方面几乎无出其右者,然而并不能识别和高亮 vimscript。即使对于其他可识别的代码,导出结果也是标签套标签、css 摞 | ||
26 | css,导出之后无论是修改代码还是更换高亮样式,蜀道之难难于上青天,除非重新导出。</p> | ||
27 | <p>这几件事便成了我为本站建立代码高亮的源动力。</p> | ||
28 | <p>过程的繁琐自不必说,让我们进入正题,<strong>用炫丽的色彩色彩点燃我们的码农世界吧</strong>!</p> | ||
29 | <p>需要注意的是,下文所有 html 使用的链接都是我使用的 url,我大部分替换为了<code>/path/to/……</code>,意在说明这里可以替换为自己实际的 url(官网链接、cdn | ||
30 | 链接、本地链接等),如有遗漏可自行修改。</p> | ||
31 | <h1 id="代码高亮">代码高亮</h1> | ||
32 | <p>现在很多网站的代码高亮插件都是<a href="https://highlightjs.org/">highlight.js</a>,其中文文档<a | ||
33 | href="http://highlight.cndoc.wiki/doc">在此</a>。该插件目前已支持近 200 种编程语言、有 75 套渲染样式,十分完备易用。利用 Vue、npm | ||
34 | 等高端技术的方法就不说了,<del>主要是我也不会</del>,就说说最简单的用法——直接在页面中加入 highlight.js 实现代码高亮。</p> | ||
35 | <p>使用方法极为简单:</p> | ||
16 | <ul> | 36 | <ul> |
17 | <li><a href="#代码高亮">代码高亮</a></li> | 37 | <li>在 html 文档的 head 中直接引入 highlight.min.js 和自己喜欢的 style 的链接 |
18 | <li><a href="#为高亮代码添加行号">为高亮代码添加行号</a></li> | ||
19 | <li><a href="#一键复制">一键复制</a> | ||
20 | <ul> | 38 | <ul> |
21 | <li><a href="#剪贴板有关">剪贴板有关</a></li> | 39 | <li>支持的 style 列表和代码语言在<a href="https://highlightjs.org/static/demo/">官方 |
22 | <li><a href="#修改pre">修改<pre></a></li> | 40 | demo</a>中,可以直接下载整套代码,也可以在 |
23 | <li><a href="#功能实现">功能实现</a></li> | 41 | F12 中只选择自己喜欢的 style 搞下来</li> |
24 | <li><a href="#教程所忘记的">教程所忘记的</a></li> | ||
25 | </ul> | 42 | </ul> |
26 | </li> | 43 | </li> |
27 | <li><a href="#抽成单独的文件">抽成单独的文件</a> | 44 | <li>将代码块用<code><pre><code>……</code></pre></code>包裹</li> |
28 | <ul> | 45 | <li>code 可以选择用 class 来标注语言,如<code><code class="language-c++">……</code></code>,也可以不标注,会自动识别语言 |
29 | <li><a href="#需要额外的css">需要额外的css</a></li> | ||
30 | <li><a href="#引入css">引入css</a></li> | ||
31 | <li><a href="#引入script">引入script</a></li> | ||
32 | <li><a href="#执行代码">执行代码</a></li> | ||
33 | </ul> | ||
34 | </li> | 46 | </li> |
35 | <li><a href="#综合">综合</a></li> | 47 | <li>代码第一行与<code><code></code>标签同行,最后一行可与<code></code></code>不同行,<strong>否则会在第一行之前出现一个空的第 0 |
36 | <li><a href="#还缺点什么吗">还缺点什么吗</a> | 48 | 行,代码块不对称</strong></li> |
37 | <ul> | ||
38 | <li><a href="#Markdown">Markdown</a></li> | ||
39 | <li><a href="#小脚本">小脚本</a></li> | ||
40 | </ul> | ||
41 | </li> | ||
42 | <li><a href="#碎碎念">碎碎念</a></li> | ||
43 | </ul> | 49 | </ul> |
44 | </span> | 50 | <pre><code><html> |
45 | <p>作为一名计算机学生,代码是生活里必不可少的组件。因而自年初开始学习前端、购买域名建站的时候,我就曾想过如何在网页里实现代码高亮。奈何才疏学浅,就此搁置。</p> | ||
46 | <p>上个月在搜索如何用代码自行实现 git 基本功能的时候,误打误撞进了大佬 Nikita 的<a href="https://www.leshenko.net/p/ugit/#">μgit | ||
47 | 教程</a>,惊为天人,不仅内容详细,排版也极为美观,尤其是代码的 git diff 显示,可以清晰地看到每一步做了哪些改变,仿佛大佬亲自莅临指导。于是在开启 μgit | ||
48 | 学习(<del>尚未有所推进,笑死</del>,算是个坑,<strong>以后补吧</strong>)的同时,与黄佬共同学习该教程所用到的代码高亮办法,发现是一个 js 插件 | ||
49 | highlight.min.js,探索过程中又找到了一个在此基础上以表格办法为代码添加行号的 js 插件 highlightjs-line-numbers.js,形成了一份草创的为 git 仓库提供 diff | ||
50 | 显示的<a | ||
51 | href="https://www.qin-juan-ge-zhu.top/code/others'code/highlight.html">网页</a>,按照目前的理解尚需一些其他工具,因此该网页几乎等同于完全不会,只能说“欲知后事如何,且听下回分解”了。(一段话两个坑?挖坑小能手实锤!) | ||
52 | </p> | ||
53 | <p>同时,一个多月来我为服务器 vim 做的配置 vimrc 变化较大。万一身边有人学呢?或者,年底服务期到期了这些设置总需要有个备份吧?故而近来想着重新写一份最新的。于是我碰到了之前无法解决的问题:之前 Markdown | ||
54 | 导出为 html 所使用的工具是 typora,在导出方面几乎无出其右者,然而并不能识别和高亮 vimscript。即使对于其他可识别的代码,导出结果也是标签套标签、css 摞 | ||
55 | css,导出之后无论是修改代码还是更换高亮样式,蜀道之难难于上青天,除非重新导出。</p> | ||
56 | <p>这几件事便成了我为本站建立代码高亮的源动力。</p> | ||
57 | <p>过程的繁琐自不必说,让我们进入正题,<strong>用炫丽的色彩色彩点燃我们的码农世界吧</strong>!</p> | ||
58 | <p>需要注意的是,下文所有 html 使用的链接都是我使用的 url,我大部分替换为了<code>/path/to/……</code>,意在说明这里可以替换为自己实际的 url(官网链接、cdn | ||
59 | 链接、本地链接等),如有遗漏可自行修改。</p> | ||
60 | <h1 id="代码高亮">代码高亮</h1> | ||
61 | <p>现在很多网站的代码高亮插件都是<a href="https://highlightjs.org/">highlight.js</a>,其中文文档<a | ||
62 | href="http://highlight.cndoc.wiki/doc">在此</a>。该插件目前已支持近 200 种编程语言、有 75 套渲染样式,十分完备易用。利用 Vue、npm | ||
63 | 等高端技术的方法就不说了,<del>主要是我也不会</del>,就说说最简单的用法——直接在页面中加入 highlight.js 实现代码高亮。</p> | ||
64 | <p>使用方法极为简单:</p> | ||
65 | <ul> | ||
66 | <li>在 html 文档的 head 中直接引入 highlight.min.js 和自己喜欢的 style 的链接 | ||
67 | <ul> | ||
68 | <li>支持的 style 列表和代码语言在<a href="https://highlightjs.org/static/demo/">官方 demo</a>中,可以直接下载整套代码,也可以在 | ||
69 | F12 中只选择自己喜欢的 style 搞下来</li> | ||
70 | </ul> | ||
71 | </li> | ||
72 | <li>将代码块用<code><pre><code>……</code></pre></code>包裹</li> | ||
73 | <li>code 可以选择用 class 来标注语言,如<code><code class="language-c++">……</code></code>,也可以不标注,会自动识别语言 | ||
74 | </li> | ||
75 | <li>代码第一行与<code><code></code>标签同行,最后一行可与<code></code></code>不同行,<strong>否则会在第一行之前出现一个空的第 0 | ||
76 | 行,代码块不对称</strong></li> | ||
77 | </ul> | ||
78 | <pre><code><html> | ||
79 | <head> | 51 | <head> |
80 | <!-- 选择最喜欢的style --> | 52 | <!-- 选择最喜欢的style --> |
81 | <link | 53 | <link |
@@ -104,14 +76,14 @@ int main() | |||
104 | </div> | 76 | </div> |
105 | </body> | 77 | </body> |
106 | </html></code></pre> | 78 | </html></code></pre> |
107 | <p>我们已经朝着代码块的最终目标迈出了一大步。</p> | 79 | <p>我们已经朝着代码块的最终目标迈出了一大步。</p> |
108 | <h1 id="为高亮代码添加行号">为高亮代码添加行号</h1> | 80 | <h1 id="为高亮代码添加行号">为高亮代码添加行号</h1> |
109 | <p>想想我们的代码编辑器,就会产生一个新诉求:为已经高亮的代码添加行号。没错,前辈大牛与我们的想法完全一致。</p> | 81 | <p>想想我们的代码编辑器,就会产生一个新诉求:为已经高亮的代码添加行号。没错,前辈大牛与我们的想法完全一致。</p> |
110 | <p>在 highlight.js 的基础上实现添加行号的有很多,这里采用的是其中一个,<a | 82 | <p>在 highlight.js 的基础上实现添加行号的有很多,这里采用的是其中一个,<a |
111 | href="https://github.com/wcoder/highlightjs-line-numbers.js/">highlightjs-line-numbers.js</a>。同样地,高端的食材往往只需要最简单的做法(努力为自己的菜找理由.jpg),这里采用直接引用 | 83 | href="https://github.com/wcoder/highlightjs-line-numbers.js/">highlightjs-line-numbers.js</a>。同样地,高端的食材往往只需要最简单的做法(努力为自己的菜找理由.jpg),这里采用直接引用 |
112 | js 的方式。</p> | 84 | js 的方式。</p> |
113 | <p>使用方法同样很简单,我们不得不佩服大佬:只需要在之前的基础上添加两行代码即可。如下:</p> | 85 | <p>使用方法同样很简单,我们不得不佩服大佬:只需要在之前的基础上添加两行代码即可。如下:</p> |
114 | <pre><code><html> | 86 | <pre><code><html> |
115 | <head> | 87 | <head> |
116 | <link | 88 | <link |
117 | rel="stylesheet" | 89 | rel="stylesheet" |
@@ -141,8 +113,8 @@ int main() | |||
141 | </div> | 113 | </div> |
142 | </body> | 114 | </body> |
143 | </html></code></pre> | 115 | </html></code></pre> |
144 | <p>在此基础上,为了使得行号进一步美观,官方甚至还给出了行号的 css,可供按需修改:</p> | 116 | <p>在此基础上,为了使得行号进一步美观,官方甚至还给出了行号的 css,可供按需修改:</p> |
145 | <pre><code>/* for block of numbers */ | 117 | <pre><code>/* for block of numbers */ |
146 | pre code td.hljs-ln-numbers { | 118 | pre code td.hljs-ln-numbers { |
147 | text-align: center; | 119 | text-align: center; |
148 | color: #9c9c9c; | 120 | color: #9c9c9c; |
@@ -158,18 +130,19 @@ pre code td.hljs-ln-code { | |||
158 | padding-left: 1rem; | 130 | padding-left: 1rem; |
159 | border: none; | 131 | border: none; |
160 | }</code></pre> | 132 | }</code></pre> |
161 | <h1 id="一键复制">一键复制</h1> | 133 | <h1 id="一键复制">一键复制</h1> |
162 | <p>代码块的功能越来越齐全,你可曾想过像万恶的 CSDN | 134 | <p>代码块的功能越来越齐全,你可曾想过像万恶的 CSDN |
163 | 搞“登录后复制”那样,来一个一键复制呢?这对于看到网站的人来说是很重要的,否则直接复制下来的代码很容易出现缩进不正确等各种情况,影响心情。<strong>说干就干!</strong></p> | 135 | 搞“登录后复制”那样,来一个一键复制呢?这对于看到网站的人来说是很重要的,否则直接复制下来的代码很容易出现缩进不正确等各种情况,影响心情。<strong>说干就干!</strong></p> |
164 | <p>可是就是这一步,难倒了我整整一天。支持 highlight.js 的一键复制不少,可是支持已经投入使用的 highlightjs-line-numbers.js | 136 | <p>可是就是这一步,难倒了我整整一天。支持 highlight.js 的一键复制不少,可是支持已经投入使用的 highlightjs-line-numbers.js |
165 | 的却不多;在此基础上,还要越简单越好,便于我这个菜鸡看懂和使用,所以像<a href="">这个</a>使用 Vue | 137 | 的却不多;在此基础上,还要越简单越好,便于我这个菜鸡看懂和使用,所以像<a href="">这个</a>使用 Vue |
166 | 还语焉不详的插件遗憾出局。到底还是做网站的心疼做网站的,最终在捣鼓了一下午和一晚上之后,深夜我终于找到了一篇<a | 138 | 还语焉不详的插件遗憾出局。到底还是做网站的心疼做网站的,最终在捣鼓了一下午和一晚上之后,深夜我终于找到了一篇<a |
167 | href="https://savalone.com/javascript/ueditor-highlightjs.html">文章</a>,与我的要求近乎完全一致,而且文章本身用的也是文章所说的代码处理办法,体验结果甚合吾意。 | 139 | href="https://savalone.com/javascript/ueditor-highlightjs.html">文章</a>,与我的要求近乎完全一致,而且文章本身用的也是文章所说的代码处理办法,体验结果甚合吾意。 |
168 | </p> | 140 | </p> |
169 | <h2 id="剪贴板有关">剪贴板有关</h2> | 141 | <h2 id="剪贴板有关">剪贴板有关</h2> |
170 | <p>一键复制与剪贴板有关。在这方面,<a href="https://github.com/zenorocha/clipboard.js">cliploard</a>是一个好用的 js。同样地,先看一下官方 Demo 吧: | 142 | <p>一键复制与剪贴板有关。在这方面,<a href="https://github.com/zenorocha/clipboard.js">cliploard</a>是一个好用的 js。同样地,先看一下官方 |
171 | </p> | 143 | Demo 吧: |
172 | <pre><code><script> | 144 | </p> |
145 | <pre><code><script> | ||
173 | <!-- 1. Define some markup --> | 146 | <!-- 1. Define some markup --> |
174 | <textarea id="bar">hello</textarea> | 147 | <textarea id="bar">hello</textarea> |
175 | <button class="btn" data-clipboard-action="cut" data-clipboard-target="#bar">Cut</button> | 148 | <button class="btn" data-clipboard-action="cut" data-clipboard-target="#bar">Cut</button> |
@@ -186,15 +159,16 @@ pre code td.hljs-ln-code { | |||
186 | console.log(e); | 159 | console.log(e); |
187 | }); | 160 | }); |
188 | </script></code></pre> | 161 | </script></code></pre> |
189 | <p>button 的 data-clipboard-action=“cut” 属性为“剪切”,data-clipboard-action=“copy” 属性为“复制”。然后可以看到 button 的 | 162 | <p>button 的 data-clipboard-action=“cut” 属性为“剪切”,data-clipboard-action=“copy” 属性为“复制”。然后可以看到 button 的 |
190 | data-clipboard-target 属性值 = 要进行复制的目标元素的 id。</p> | 163 | data-clipboard-target 属性值 = 要进行复制的目标元素的 id。</p> |
191 | <p>那么我们就要在每一个 pre 里面添加一个 data-clipboard-target 属性不同的 button 按钮,并且要为这个 pre 增加一个和 button 的 data-clipboard-target | 164 | <p>那么我们就要在每一个 pre 里面添加一个 data-clipboard-target 属性不同的 button 按钮,并且要为这个 pre 增加一个和 button 的 |
192 | 属性一样的 id,最后调用 demo 里的 js。<strong>前进!</strong></p> | 165 | data-clipboard-target |
193 | <h2 id="修改pre">修改<code><pre></code></h2> | 166 | 属性一样的 id,最后调用 demo 里的 js。<strong>前进!</strong></p> |
194 | <p>首先,引入 clipboard.js:</p> | 167 | <h2 id="修改pre">修改<code><pre></code></h2> |
195 | <pre><code><script src="url/clipboard.min.js"></script></code></pre> | 168 | <p>首先,引入 clipboard.js:</p> |
196 | <p>为每一个 pre 添加复制按钮:</p> | 169 | <pre><code><script src="url/clipboard.min.js"></script></code></pre> |
197 | <pre><code><script type="text/javascript"> | 170 | <p>为每一个 pre 添加复制按钮:</p> |
171 | <pre><code><script type="text/javascript"> | ||
198 | var allpre = document.getElementsByTagName("pre"); | 172 | var allpre = document.getElementsByTagName("pre"); |
199 | for (i = 0; i < allpre.length; i++) { | 173 | for (i = 0; i < allpre.length; i++) { |
200 | var onepre = document.getElementsByTagName("pre")[i]; | 174 | var onepre = document.getElementsByTagName("pre")[i]; |
@@ -207,22 +181,22 @@ pre code td.hljs-ln-code { | |||
207 | mycode; | 181 | mycode; |
208 | } | 182 | } |
209 | </script></code></pre> | 183 | </script></code></pre> |
210 | <p>为 pre 循环增加 id,并为 pre 内的 button 增加 data-clipboard-target 属性:</p> | 184 | <p>为 pre 循环增加 id,并为 pre 内的 button 增加 data-clipboard-target 属性:</p> |
211 | <pre><code><script> | 185 | <pre><code><script> |
212 | $("pre").each(function () { | 186 | $("pre").each(function () { |
213 | $(this).attr("id", "pre" + $(this).index()); | 187 | $(this).attr("id", "pre" + $(this).index()); |
214 | var btns = $(this).find("button"); | 188 | var btns = $(this).find("button"); |
215 | $(btns).attr("data-clipboard-target", "#pre" + $(this).index()); | 189 | $(btns).attr("data-clipboard-target", "#pre" + $(this).index()); |
216 | }); | 190 | }); |
217 | </script></code></pre> | 191 | </script></code></pre> |
218 | <h2 id="功能实现">功能实现</h2> | 192 | <h2 id="功能实现">功能实现</h2> |
219 | <p>为了简单起见,我们可以将之前的代码高亮与行号放在这里,也就是上边的各个 script 之后:</p> | 193 | <p>为了简单起见,我们可以将之前的代码高亮与行号放在这里,也就是上边的各个 script 之后:</p> |
220 | <pre><code><script> | 194 | <pre><code><script> |
221 | hljs.highlightAll(); | 195 | hljs.highlightAll(); |
222 | hljs.initLineNumbersOnLoad(); | 196 | hljs.initLineNumbersOnLoad(); |
223 | </script></code></pre> | 197 | </script></code></pre> |
224 | <p>而后直接复制 demo 目录内 target-textarea.html 的 js 代码:</p> | 198 | <p>而后直接复制 demo 目录内 target-textarea.html 的 js 代码:</p> |
225 | <pre><code><!--这个是官方 demo 的用法--> | 199 | <pre><code><!--这个是官方 demo 的用法--> |
226 | <script> | 200 | <script> |
227 | var clipboard = new ClipboardJS(".btn"); | 201 | var clipboard = new ClipboardJS(".btn"); |
228 | clipboard.on("success", function (e) { | 202 | clipboard.on("success", function (e) { |
@@ -232,8 +206,8 @@ pre code td.hljs-ln-code { | |||
232 | console.log(e); | 206 | console.log(e); |
233 | }); | 207 | }); |
234 | </script></code></pre> | 208 | </script></code></pre> |
235 | <p>原教程中提供的 css:</p> | 209 | <p>原教程中提供的 css:</p> |
236 | <pre><code>pre { | 210 | <pre><code>pre { |
237 | position: relative; | 211 | position: relative; |
238 | } | 212 | } |
239 | pre .btn { | 213 | pre .btn { |
@@ -251,10 +225,10 @@ pre .btn:hover { | |||
251 | pre:hover .btn { | 225 | pre:hover .btn { |
252 | display: block; | 226 | display: block; |
253 | }</code></pre> | 227 | }</code></pre> |
254 | <h2 id="教程所忘记的">教程所忘记的</h2> | 228 | <h2 id="教程所忘记的">教程所忘记的</h2> |
255 | <p>除此之外,由于一键复制的某些代码与前边关联,且需要足够的时间等待文档加载结束再执行,因而所有上述功能代码需要放置在文档末尾。</p> | 229 | <p>除此之外,由于一键复制的某些代码与前边关联,且需要足够的时间等待文档加载结束再执行,因而所有上述功能代码需要放置在文档末尾。</p> |
256 | <p>就好了?我做到这的时候也这么想。可惜事与愿违,如果我们仅仅是按照上边已经说的去写开头的那份 Hello World,我们会得到如下一篇文档。</p> | 230 | <p>就好了?我做到这的时候也这么想。可惜事与愿违,如果我们仅仅是按照上边已经说的去写开头的那份 Hello World,我们会得到如下一篇文档。</p> |
257 | <pre><code><html> | 231 | <pre><code><html> |
258 | <head> | 232 | <head> |
259 | <!-- style css引入 --> | 233 | <!-- style css引入 --> |
260 | <link | 234 | <link |
@@ -353,16 +327,16 @@ int main() | |||
353 | }); | 327 | }); |
354 | </script> | 328 | </script> |
355 | </html></code></pre> | 329 | </html></code></pre> |
356 | <p>直接打开浏览器看效果,会发现如下几个问题:</p> | 330 | <p>直接打开浏览器看效果,会发现如下几个问题:</p> |
357 | <ul> | 331 | <ul> |
358 | <li>“复制”与“已复制”同时出现,且后者被前者遮盖,点击“复制”时候也会出现但还是被遮盖;而我们需要的是一般情况“复制”,复制过了才“已复制”</li> | 332 | <li>“复制”与“已复制”同时出现,且后者被前者遮盖,点击“复制”时候也会出现但还是被遮盖;而我们需要的是一般情况“复制”,复制过了才“已复制”</li> |
359 | <li>点击“复制”的时候,复制到的东西把“复制”这两个字也复制进去了</li> | 333 | <li>点击“复制”的时候,复制到的东西把“复制”这两个字也复制进去了</li> |
360 | <li>复制的时候所有内容会被全选中,一般默认是蓝底白字,十分有碍观瞻,我们需要的是点击“复制”的时候代码展示没有变化(选不选中的,看不见就行了)</li> | 334 | <li>复制的时候所有内容会被全选中,一般默认是蓝底白字,十分有碍观瞻,我们需要的是点击“复制”的时候代码展示没有变化(选不选中的,看不见就行了)</li> |
361 | <li>“复制”与“已复制”字体大小、按钮大小均不一致</li> | 335 | <li>“复制”与“已复制”字体大小、按钮大小均不一致</li> |
362 | <li>还一个啥问题来着?忘了,最近脑子不好使,似乎想闹独立。</li> | 336 | <li>还一个啥问题来着?忘了,最近脑子不好使,似乎想闹独立。</li> |
363 | </ul> | 337 | </ul> |
364 | <p>总而言之,教程内容里差点东西。由于教程本身使用的也是这一套代码,F12 理所当然地能解决一切。最后我发现,作者漏了几段重要的 css:</p> | 338 | <p>总而言之,教程内容里差点东西。由于教程本身使用的也是这一套代码,F12 理所当然地能解决一切。最后我发现,作者漏了几段重要的 css:</p> |
365 | <pre><code>pre .btn-tip { | 339 | <pre><code>pre .btn-tip { |
366 | display: none; | 340 | display: none; |
367 | position: absolute; | 341 | position: absolute; |
368 | top: 0px; | 342 | top: 0px; |
@@ -405,26 +379,27 @@ int main() | |||
405 | code ::selection { | 379 | code ::selection { |
406 | background-color: rgba(0, 0, 0, 0); | 380 | background-color: rgba(0, 0, 0, 0); |
407 | }</code></pre> | 381 | }</code></pre> |
408 | <p>除了加了注释说明功能的地方之外,剩的是一些不甚打紧的参数,而且效果也比较满意,因此不需要修改,直接将上述 css 加入 style 标签内就好了。此时再打开这个 html,Hello World | 382 | <p>除了加了注释说明功能的地方之外,剩的是一些不甚打紧的参数,而且效果也比较满意,因此不需要修改,直接将上述 css 加入 style 标签内就好了。此时再打开这个 html,Hello World |
409 | 已经高亮,带有行号和一键复制,使用体验丝滑流畅。一切都是那么和谐,仿佛自来就是如此,什么都未发生过。</p> | 383 | 已经高亮,带有行号和一键复制,使用体验丝滑流畅。一切都是那么和谐,仿佛自来就是如此,什么都未发生过。</p> |
410 | <p>历尽千辛万苦,我们即将来到旅程的终点。</p> | 384 | <p>历尽千辛万苦,我们即将来到旅程的终点。</p> |
411 | <p><em><strong>吗?</strong></em></p> | 385 | <p><em><strong>吗?</strong></em></p> |
412 | <h1 id="抽成单独的文件">抽成单独的文件</h1> | 386 | <h1 id="抽成单独的文件">抽成单独的文件</h1> |
413 | <p>样例程序已经达到预期效果,但为了做网站,我们的目标远不止于此。我们需要的是<strong>将所需内容尽可能简单地引入已经写好的文档</strong>(我一般是用 pandoc/Typora 将 Markdown 导出为 | 387 | <p>样例程序已经达到预期效果,但为了做网站,我们的目标远不止于此。我们需要的是<strong>将所需内容尽可能简单地引入已经写好的文档</strong>(我一般是用 pandoc/Typora 将 Markdown |
414 | html),最好只是一个文件。没错,我们最后要做的,是将上边已经有的 script 标签、link 标签、style 标签和 js 代码写在同一份 js 文件里,只需要在 head 中引用该文件即可。</p> | 388 | 导出为 |
415 | <p>首先,记录一下现有的要求:</p> | 389 | html),最好只是一个文件。没错,我们最后要做的,是将上边已经有的 script 标签、link 标签、style 标签和 js 代码写在同一份 js 文件里,只需要在 head 中引用该文件即可。</p> |
416 | <ul> | 390 | <p>首先,记录一下现有的要求:</p> |
417 | <li>head 中的标签 | 391 | <ul> |
418 | <ul> | 392 | <li>head 中的标签 |
419 | <li>需要引入 style 样式,可以考虑加随机 style</li> | 393 | <ul> |
420 | <li>jquery 需要在其他的 script 之前引入,highlight.js 要在 highlightjs-line-number 之前引入,clipboard 最后引入</li> | 394 | <li>需要引入 style 样式,可以考虑加随机 style</li> |
421 | <li>所有要执行的 js 代码必须在文档整体完成之后再执行</li> | 395 | <li>jquery 需要在其他的 script 之前引入,highlight.js 要在 highlightjs-line-number 之前引入,clipboard 最后引入</li> |
422 | </ul> | 396 | <li>所有要执行的 js 代码必须在文档整体完成之后再执行</li> |
423 | </li> | 397 | </ul> |
424 | </ul> | 398 | </li> |
425 | <p>经过多次试验和黄四郎的悉心指导,得到了以下的方法。</p> | 399 | </ul> |
426 | <h2 id="需要的额外-css">需要的额外 css</h2> | 400 | <p>经过多次试验和黄四郎的悉心指导,得到了以下的方法。</p> |
427 | <pre><code>/* highlight.css */ | 401 | <h2 id="需要的额外-css">需要的额外 css</h2> |
402 | <pre><code>/* highlight.css */ | ||
428 | 403 | ||
429 | /* for block of numbers */ | 404 | /* for block of numbers */ |
430 | pre code td.hljs-ln-numbers { | 405 | pre code td.hljs-ln-numbers { |
@@ -509,20 +484,20 @@ pre .btn-tip { | |||
509 | code ::selection { | 484 | code ::selection { |
510 | background-color: rgba(0, 0, 0, 0); | 485 | background-color: rgba(0, 0, 0, 0); |
511 | }</code></pre> | 486 | }</code></pre> |
512 | <h2 id="引入-css">引入 css</h2> | 487 | <h2 id="引入-css">引入 css</h2> |
513 | <p>首先写一个为文档引入 css 的 js 函数:</p> | 488 | <p>首先写一个为文档引入 css 的 js 函数:</p> |
514 | <pre><code>function addLink(href, rel = "stylesheet") { | 489 | <pre><code>function addLink(href, rel = "stylesheet") { |
515 | // 默认为stylesheet | 490 | // 默认为stylesheet |
516 | const link = document.createElement("link"); | 491 | const link = document.createElement("link"); |
517 | link.rel = rel; | 492 | link.rel = rel; |
518 | link.href = href; | 493 | link.href = href; |
519 | document.head.appendChild(link); | 494 | document.head.appendChild(link); |
520 | }</code></pre> | 495 | }</code></pre> |
521 | <p>而后引入上述的额外 css:</p> | 496 | <p>而后引入上述的额外 css:</p> |
522 | <pre><code>// 加载高亮所需特殊css | 497 | <pre><code>// 加载高亮所需特殊css |
523 | addLink("/path/to/highlight.css");</code></pre> | 498 | addLink("/path/to/highlight.css");</code></pre> |
524 | <p>还需要选择高亮 style 样式。不知道你是否和我一样有选择困难症?每次写一篇文档都去选一个 style 着实很烦。不如随机抽取:</p> | 499 | <p>还需要选择高亮 style 样式。不知道你是否和我一样有选择困难症?每次写一篇文档都去选一个 style 着实很烦。不如随机抽取:</p> |
525 | <pre><code>// 此处负责随机选择一个代码高亮style的css并加载 | 500 | <pre><code>// 此处负责随机选择一个代码高亮style的css并加载 |
526 | // 数组内是现highlight.js所有插件名称,共计75个 | 501 | // 数组内是现highlight.js所有插件名称,共计75个 |
527 | const styles = [ | 502 | const styles = [ |
528 | "a11y-dark.min.css", | 503 | "a11y-dark.min.css", |
@@ -604,10 +579,10 @@ const styles = [ | |||
604 | 579 | ||
605 | const href = "/path/to/styles/" + styles[~~(Math.random() * styles.length)]; | 580 | const href = "/path/to/styles/" + styles[~~(Math.random() * styles.length)]; |
606 | addLink(href);</code></pre> | 581 | addLink(href);</code></pre> |
607 | <p>于是我们完成了 style 的高亮 style 随机选择。</p> | 582 | <p>于是我们完成了 style 的高亮 style 随机选择。</p> |
608 | <h2 id="引入-script">引入 script</h2> | 583 | <h2 id="引入-script">引入 script</h2> |
609 | <p>老规矩,先写一个引入 script 的函数:</p> | 584 | <p>老规矩,先写一个引入 script 的函数:</p> |
610 | <pre><code>// 加js的函数 | 585 | <pre><code>// 加js的函数 |
611 | function addScript(url, async = true) { | 586 | function addScript(url, async = true) { |
612 | //默认先到先得地加载 | 587 | //默认先到先得地加载 |
613 | const script = document.createElement("script"); | 588 | const script = document.createElement("script"); |
@@ -615,9 +590,9 @@ function addScript(url, async = true) { | |||
615 | script.async = async; | 590 | script.async = async; |
616 | document.head.appendChild(script); | 591 | document.head.appendChild(script); |
617 | }</code></pre> | 592 | }</code></pre> |
618 | <p>接下来按照上边说过的顺序(先 jQuery,再 highlight.min.js,而后 highlightjs-line-numbers.js,最后 | 593 | <p>接下来按照上边说过的顺序(先 jQuery,再 highlight.min.js,而后 highlightjs-line-numbers.js,最后 |
619 | clipboard.js)进行加载。为了防止浏览器并发地加载,<strong>必须规定<code>async=false</code></strong>。</p> | 594 | clipboard.js)进行加载。为了防止浏览器并发地加载,<strong>必须规定<code>async=false</code></strong>。</p> |
620 | <pre><code>// 此处是各个插件 | 595 | <pre><code>// 此处是各个插件 |
621 | // 首先放置最高的jQuery备用 | 596 | // 首先放置最高的jQuery备用 |
622 | // 而后是highlight.min.js高亮代码 | 597 | // 而后是highlight.min.js高亮代码 |
623 | // 再加行号 | 598 | // 再加行号 |
@@ -633,9 +608,9 @@ for (let i in url) { | |||
633 | // 要求必须按顺序加载 | 608 | // 要求必须按顺序加载 |
634 | addScript(url[i], false); | 609 | addScript(url[i], false); |
635 | }</code></pre> | 610 | }</code></pre> |
636 | <h2 id="执行代码">执行代码</h2> | 611 | <h2 id="执行代码">执行代码</h2> |
637 | <p>最后、也是最重要的部分,当然是要在文档末尾才执行的代码啦~所以我们要在原功能代码的基础上套一层<code>onload()</code>,如下:</p> | 612 | <p>最后、也是最重要的部分,当然是要在文档末尾才执行的代码啦~所以我们要在原功能代码的基础上套一层<code>onload()</code>,如下:</p> |
638 | <pre><code>// 以下代码负责使用上述模块,因而必须在html文档整体完成之后加载 | 613 | <pre><code>// 以下代码负责使用上述模块,因而必须在html文档整体完成之后加载 |
639 | onload = () => { | 614 | onload = () => { |
640 | // 导出的源码由<pre><code>……</code></pre>包裹,符合highlight.min.js要求 | 615 | // 导出的源码由<pre><code>……</code></pre>包裹,符合highlight.min.js要求 |
641 | // 需要在code之前加入复制按钮,要求点击之后显示已复制 | 616 | // 需要在code之前加入复制按钮,要求点击之后显示已复制 |
@@ -673,9 +648,9 @@ onload = () => { | |||
673 | console.log(e); | 648 | console.log(e); |
674 | }); | 649 | }); |
675 | };</code></pre> | 650 | };</code></pre> |
676 | <h1 id="综合">综合</h1> | 651 | <h1 id="综合">综合</h1> |
677 | <p>总而言之,现在我们得到了两份文件:</p> | 652 | <p>总而言之,现在我们得到了两份文件:</p> |
678 | <pre><code>/* highlight.css */ | 653 | <pre><code>/* highlight.css */ |
679 | 654 | ||
680 | /* for block of numbers */ | 655 | /* for block of numbers */ |
681 | pre code td.hljs-ln-numbers { | 656 | pre code td.hljs-ln-numbers { |
@@ -760,8 +735,8 @@ pre .btn-tip { | |||
760 | code ::selection { | 735 | code ::selection { |
761 | background-color: rgba(0, 0, 0, 0); | 736 | background-color: rgba(0, 0, 0, 0); |
762 | }</code></pre> | 737 | }</code></pre> |
763 | <p>还有一份 js:</p> | 738 | <p>还有一份 js:</p> |
764 | <pre><code>// myhighlight.js | 739 | <pre><code>// myhighlight.js |
765 | // 代码高亮、加行号、加一键复制功能 | 740 | // 代码高亮、加行号、加一键复制功能 |
766 | 741 | ||
767 | // 加js的函数 | 742 | // 加js的函数 |
@@ -923,8 +898,8 @@ onload = () => { | |||
923 | console.log(e); | 898 | console.log(e); |
924 | }); | 899 | }); |
925 | };</code></pre> | 900 | };</code></pre> |
926 | <p>使用方法也很简单,只需要引入上边这个 myhighlight.js 即可。比如之前写好的 Hello World 页面:</p> | 901 | <p>使用方法也很简单,只需要引入上边这个 myhighlight.js 即可。比如之前写好的 Hello World 页面:</p> |
927 | <pre><code><html> | 902 | <pre><code><html> |
928 | <head> | 903 | <head> |
929 | <title>Hello World展示</title> | 904 | <title>Hello World展示</title> |
930 | <!-- 前边需要什么css/js还可以自己添 --> | 905 | <!-- 前边需要什么css/js还可以自己添 --> |
@@ -947,23 +922,23 @@ int main() | |||
947 | </div> | 922 | </div> |
948 | </body> | 923 | </body> |
949 | </html></code></pre> | 924 | </html></code></pre> |
950 | <p><strong>代码多跑路,<kbd>Ctrl+c/v</kbd>少跑腿,国安民乐,岂不美哉?</strong></p> | 925 | <p><strong>代码多跑路,<kbd>Ctrl+c/v</kbd>少跑腿,国安民乐,岂不美哉?</strong></p> |
951 | <h1 id="还缺点什么吗">还缺点什么吗?</h1> | 926 | <h1 id="还缺点什么吗">还缺点什么吗?</h1> |
952 | <p>“还缺?这不已经好了吗?”没错,代码部分好了,但对我们而言差点东西。</p> | 927 | <p>“还缺?这不已经好了吗?”没错,代码部分好了,但对我们而言差点东西。</p> |
953 | <h2 id="markdown">Markdown</h2> | 928 | <h2 id="markdown">Markdown</h2> |
954 | <p>根据上边已经有的代码高亮,现在完成手搓 html 写网页已经很轻松加愉快了,常用的无非就是那几个标签罢了。为了更简单一些,难道不能写 Markdown 吗?当然可以!</p> | 929 | <p>根据上边已经有的代码高亮,现在完成手搓 html 写网页已经很轻松加愉快了,常用的无非就是那几个标签罢了。为了更简单一些,难道不能写 Markdown 吗?当然可以!</p> |
955 | <p>有两个特别好的小工具,一个叫<a href="https://typora.io/">Typora</a>,另一个叫<a href="https://pandoc.org/">pandoc</a>。前者是专门的 | 930 | <p>有两个特别好的小工具,一个叫<a href="https://typora.io/">Typora</a>,另一个叫<a href="https://pandoc.org/">pandoc</a>。前者是专门的 |
956 | Markdown 编辑器,可以做到实时预览、所见即所得,主题样式也可以自由选择和安装;后者则是一个做文档转换的命令行小工具。二者都可以实现 Markdown 转 html。笔者目前是在 Linux | 931 | Markdown 编辑器,可以做到实时预览、所见即所得,主题样式也可以自由选择和安装;后者则是一个做文档转换的命令行小工具。二者都可以实现 Markdown 转 html。笔者目前是在 Linux |
957 | 里写这篇文章,因而现在用的是 pandoc。</p> | 932 | 里写这篇文章,因而现在用的是 pandoc。</p> |
958 | <p>有几个点需要注意:</p> | 933 | <p>有几个点需要注意:</p> |
959 | <ul> | 934 | <ul> |
960 | <li>Typora 是买断制收费,刚下载会有 15 天试用期,如需购买则目前是$14.99(合¥ 89.00),一个账号最多可以同时在三个口使用(也就是一个号最多能供三个人用,可以合买)</li> | 935 | <li>Typora 是买断制收费,刚下载会有 15 天试用期,如需购买则目前是$14.99(合¥ 89.00),一个账号最多可以同时在三个口使用(也就是一个号最多能供三个人用,可以合买)</li> |
961 | <li>Typora 一部分导出功能需要 pandoc 支持</li> | 936 | <li>Typora 一部分导出功能需要 pandoc 支持</li> |
962 | <li>pandoc 在各种文件类型(包括 Markdown 在内)导出为 pdf 时需要 LaTeX 支持,因而如果没有 LaTeX 那就不要导出为 pdf。</li> | 937 | <li>pandoc 在各种文件类型(包括 Markdown 在内)导出为 pdf 时需要 LaTeX 支持,因而如果没有 LaTeX 那就不要导出为 pdf。</li> |
963 | <li>使用上述两种工具导出 Markdown 时,一般的代码会自带高亮,但是此种高亮是由 css 和标签一层一层摞起来的,不便于更改代码本身和代码样式,更不可能实现上述的一键复制。因而笔者建议,在编写 | 938 | <li>使用上述两种工具导出 Markdown 时,一般的代码会自带高亮,但是此种高亮是由 css 和标签一层一层摞起来的,不便于更改代码本身和代码样式,更不可能实现上述的一键复制。因而笔者建议,在编写 |
964 | Markdown 时候可以标明代码块用的语言,导出之前全部删掉。如:</li> | 939 | Markdown 时候可以标明代码块用的语言,导出之前全部删掉。如:</li> |
965 | </ul> | 940 | </ul> |
966 | <pre><code><!-- 下边用三个`作代码块示例好像不识别,改用···,请注意 --> | 941 | <pre><code><!-- 下边用三个`作代码块示例好像不识别,改用···,请注意 --> |
967 | 942 | ||
968 | <!-- 写的时候 --> | 943 | <!-- 写的时候 --> |
969 | 944 | ||
@@ -976,9 +951,9 @@ int main() | |||
976 | ··· | 951 | ··· |
977 | #include <iostream> | 952 | #include <iostream> |
978 | ···</code></pre> | 953 | ···</code></pre> |
979 | <h2 id="小脚本">小脚本</h2> | 954 | <h2 id="小脚本">小脚本</h2> |
980 | <p>pandoc 导出十分方便,要改造为我们所需的 html 还需要一些批量处理。这里我写了两个脚本,按需取用。</p> | 955 | <p>pandoc 导出十分方便,要改造为我们所需的 html 还需要一些批量处理。这里我写了两个脚本,按需取用。</p> |
981 | <pre><code># vime.sh | 956 | <pre><code># vime.sh |
982 | # 本文件用来在命令行里执行vim的文件操作 | 957 | # 本文件用来在命令行里执行vim的文件操作 |
983 | #!/bin/bash | 958 | #!/bin/bash |
984 | 959 | ||
@@ -989,8 +964,8 @@ echo $2 | |||
989 | read -n 1 | 964 | read -n 1 |
990 | # 执行 | 965 | # 执行 |
991 | find -name "$1" -exec sed -i "$2" {} \;</code></pre> | 966 | find -name "$1" -exec sed -i "$2" {} \;</code></pre> |
992 | <p>另一个脚本:</p> | 967 | <p>另一个脚本:</p> |
993 | <pre><code># pan4html.sh | 968 | <pre><code># pan4html.sh |
994 | #!/bin/bash | 969 | #!/bin/bash |
995 | 970 | ||
996 | # 这一行原想删除导出文件的style标签,好像没起作用 | 971 | # 这一行原想删除导出文件的style标签,好像没起作用 |
@@ -1008,13 +983,14 @@ vime $1 "s/<\/body>/<\/div><\/body>/g" | |||
1008 | # 从而实现表格居中、表内文字左对齐 | 983 | # 从而实现表格居中、表内文字左对齐 |
1009 | vime $1 "s/<table>/<div class=\"table-div\"><table>/g" | 984 | vime $1 "s/<table>/<div class=\"table-div\"><table>/g" |
1010 | vime $1 "s/<\/table>/<\/table><\/div>/g"</code></pre> | 985 | vime $1 "s/<\/table>/<\/table><\/div>/g"</code></pre> |
1011 | <h1 id="碎碎念">碎碎念</h1> | 986 | <h1 id="碎碎念">碎碎念</h1> |
1012 | <p>看起来文章内容很长,实际核心内容很短。核心内容很短,但探索的经历很长。</p> | 987 | <p>看起来文章内容很长,实际核心内容很短。核心内容很短,但探索的经历很长。</p> |
1013 | <p>开头已经说过,之所以需要做代码高亮、行号、一键复制功能,是因为 pandoc 和 Typora 这两个我常用的 Markdown 导出工具均不支持 vimscript | 988 | <p>开头已经说过,之所以需要做代码高亮、行号、一键复制功能,是因为 pandoc 和 Typora 这两个我常用的 Markdown 导出工具均不支持 vimscript |
1014 | 高亮。二者实际上在导出方面做的非常好,绝大多数语言都支持高亮,审美也相当不错,但是由于这一点问题,居然使我弄出了一键复制,也还是很令我自己吃惊的。</p> | 989 | 高亮。二者实际上在导出方面做的非常好,绝大多数语言都支持高亮,审美也相当不错,但是由于这一点问题,居然使我弄出了一键复制,也还是很令我自己吃惊的。</p> |
1015 | <p>现在代码块的功能可云大备,也许以后大多数文章都会改用这一套代码高亮吧。大概是不会再改了。</p> | 990 | <p>现在代码块的功能可云大备,也许以后大多数文章都会改用这一套代码高亮吧。大概是不会再改了。</p> |
1016 | <p class="time">2023.7.8</p> | 991 | <p class="time">2023.7.8</p> |
1017 | <script src="https://www.qin-juan-ge-zhu.top/common/js/comment.js"></script> | 992 | <script src="https://www.qin-juan-ge-zhu.top/common/js/comment.js"></script> |
993 | </div> | ||
1018 | </div> | 994 | </div> |
1019 | </body> | 995 | </body> |
1020 | 996 | ||
diff --git a/code/linux/aliyun_cloud.html b/code/linux/aliyun_cloud.html index 4851312..dd1e429 100644 --- a/code/linux/aliyun_cloud.html +++ b/code/linux/aliyun_cloud.html | |||
@@ -3,6 +3,7 @@ | |||
3 | 3 | ||
4 | <head> | 4 | <head> |
5 | <meta charset="utf-8" /> | 5 | <meta charset="utf-8" /> |
6 | <meta name="generator" content="pandoc" /> | ||
6 | <meta name="viewport" content="width=device-width, initial-scale=1.0, user-scalable=yes" /> | 7 | <meta name="viewport" content="width=device-width, initial-scale=1.0, user-scalable=yes" /> |
7 | <title>我的免费云服</title> | 8 | <title>我的免费云服</title> |
8 | <link rel="stylesheet" href="https://www.qin-juan-ge-zhu.top/common/CSS/pandoc.css"> | 9 | <link rel="stylesheet" href="https://www.qin-juan-ge-zhu.top/common/CSS/pandoc.css"> |
@@ -30,16 +31,15 @@ | |||
30 | <p>实验学习过程中会发给一个临时账号,自动创建一个临时的实例(服务器)供实验之用,时长 1h。讲真,阿里云的实验学习做的非常好,实验文档就在实验界面旁边,非常方便。B | 31 | <p>实验学习过程中会发给一个临时账号,自动创建一个临时的实例(服务器)供实验之用,时长 1h。讲真,阿里云的实验学习做的非常好,实验文档就在实验界面旁边,非常方便。B |
31 | 站/油管/知乎等诸多平台都有这个实验的详细教程可以参考。</p> | 32 | 站/油管/知乎等诸多平台都有这个实验的详细教程可以参考。</p> |
32 | <p>另外,由于是一个临时账号,我个人建议另找一个浏览器做实验,方便和自己现在用的帐户/ECS 实例区别开来。</p> | 33 | <p>另外,由于是一个临时账号,我个人建议另找一个浏览器做实验,方便和自己现在用的帐户/ECS 实例区别开来。</p> |
33 | <h1 id="服器配置">服器配置</h1> | 34 | <h2 id="件配置">件配置</h2> |
34 | <p>服务器硬件配置应该都是固定的吧?CPU 双核,内存 2G,硬盘 40G,带宽 1MBps。我个人觉得这个配置还是挺不错的,毕竟是白嫖,要求也不能太高了不是?</p> | 35 | <p>服务器硬件配置应该都是固定的吧?CPU 双核,内存 2G,硬盘 40G,带宽 1MBps。我个人觉得这个配置还是挺不错的,毕竟是白嫖,要求也不能太高了不是?</p> |
35 | <p>操作系统方面,由于个人比较喜欢纯命令行操作,所以选择了 Ubuntu22.04 LTS,我的虚拟机也是这个版本,相对而言会比其他发行版更熟悉一些。</p> | 36 | <p>操作系统方面,由于个人比较喜欢纯命令行操作,所以选择了 Ubuntu22.04 LTS,我的虚拟机也是这个版本,相对而言会比其他发行版更熟悉一些。</p> |
36 | <h2 id="服务器初始化">服务器初始化</h2> | 37 | <h1 id="服务器初始化">服务器初始化</h1> |
37 | <p>在跟着创建实例的要求一步步做好之后,我们就拥有了一台属于自己的服务器了,并已经有了根用户 root。接下来,我们需要做一些初始化的工作。</p> | 38 | <p>在跟着创建实例的要求一步步做好之后,我们就拥有了一台属于自己的服务器了,并已经有了根用户 root。接下来,我们需要做一些初始化的工作。</p> |
38 | <h3 id="创建新用户">创建新用户</h3> | 39 | <h2 id="创建新用户">创建新用户</h2> |
39 | <p>众所周知,<strong>为了安全起见,我们需要一个非 root 但具有 sudo | 40 | <p>众所周知,<strong>为了安全起见,我们需要一个非 root 但具有 sudo |
40 | 权限的用户来进行日常操作</strong>。这里我创建了一个名为<kbd>player</kbd>的用户并设置了密码,然后将其加入 | 41 | 权限的用户来进行日常操作</strong>。这里我创建了一个名为<kbd>player</kbd>的用户并设置了密码,然后将其加入 sudo 组。</p> |
41 | sudo 组。</p> | 42 | <pre><code># 创建新用户,-m参数创建用户,-d参数手动指定用户的主目录 |
42 | <pre><code class="language-bash"># 创建新用户,-m参数创建用户,-d参数手动指定用户的主目录 | ||
43 | sudo useradd -m player -d /home/player | 43 | sudo useradd -m player -d /home/player |
44 | # 如需指定默认shell,可以加上-s参数。 | 44 | # 如需指定默认shell,可以加上-s参数。 |
45 | # 这里需要注意,一般意义上bash是sh的超集,因此sh能干的bash一般都能干,所以这里一般需要指定为bash。 | 45 | # 这里需要注意,一般意义上bash是sh的超集,因此sh能干的bash一般都能干,所以这里一般需要指定为bash。 |
@@ -54,26 +54,11 @@ sudo apt install vim | |||
54 | 54 | ||
55 | cd /etc | 55 | cd /etc |
56 | sudo vim sudoers | 56 | sudo vim sudoers |
57 | # 找到root ALL=(ALL:ALL) ALL,按o新开一行,然后写一份一样的,把root改成player | 57 | # 找到root ALL=(ALL:ALL) ALL, |
58 | # 按ESC,输入:wq保存退出</code></pre> | 58 | # 新开一行然后写一份一样的,把root改成player,保存退出</code></pre> |
59 | <p>这时我们会发现一个很操蛋的事情,“为啥命令提示符之前不显示当前用户名和当前路径嘞?这岂不是每次看路径都需要<kbd>pwd</kbd>吗?烦不烦啊!”别急,问题很好修改。</p> | ||
60 | <pre><code class="language-bash">cd /home/player | ||
61 | vim .bashrc | ||
62 | # 找到PS1=,将其修改为PS1="\u@\h:\w\$" | ||
63 | # 按ESC,输入:wq保存退出</code></pre> | ||
64 | <p>修改退出之后似乎一切都没有变化。我们需要重启一下服务器。</p> | ||
65 | <pre><code>sudo reboot</code></pre> | ||
66 | <p>重新连接之后,也许还是没有变化,我就是这样的。后来在处理其它问题的时候,我发现了根本原因所在:我 TM 创建用户的时候忘了加<kbd>-s</kbd>参数,导致新用户的默认 shell | ||
67 | 是<kbd>sh</kbd>而不是<kbd>bash</kbd>。所以,我们需要修改一下新用户的默认 shell。</p> | ||
68 | <pre><code class="language-bash"># 查看当前用户的默认shell | ||
69 | echo $SHELL | ||
70 | |||
71 | # 修改默认shell | ||
72 | sudo chsh -s /bin/bash player</code></pre> | ||
73 | <p>这时再重启服务器,就会发现,一切都正常了,一切是那么美好。</p> | ||
74 | <h2 id="安装软件">安装软件</h2> | 59 | <h2 id="安装软件">安装软件</h2> |
75 | <p>经历了上边的配置,现在我们拥有了一台可以正常使用的服务器。服务器是为了什么?开发!开发!开发!所以,我们需要安装一些开发所需的软件。</p> | 60 | <p>经历了上边的配置,现在我们拥有了一台可以正常使用的服务器。服务器是为了什么?开发!开发!开发!所以,我们需要安装一些开发所需的软件。</p> |
76 | <pre><code class="language-bash"># 如需添加软件源,可以先执行下边的命令 | 61 | <pre><code># 如需添加软件源,可以先执行下边的命令 |
77 | # 备份软件源 | 62 | # 备份软件源 |
78 | sudo cp /etc/apt/sources.list /etc/apt/sources.list.old | 63 | sudo cp /etc/apt/sources.list /etc/apt/sources.list.old |
79 | # 修改软件源 | 64 | # 修改软件源 |
@@ -98,18 +83,169 @@ sudo apt install gcc | |||
98 | sudo apt install make cmake | 83 | sudo apt install make cmake |
99 | 84 | ||
100 | # 剩下需要安装的软件,可以自行搜索</code></pre> | 85 | # 剩下需要安装的软件,可以自行搜索</code></pre> |
101 | <h2 id="配置-ssh">配置 ssh</h2> | 86 | <h2 id="简单美化一下命令行">简单美化一下命令行</h2> |
87 | <p>配到这里,我们会发现一个很操蛋的事情,“为啥命令提示符之前不显示当前用户名和当前路径嘞?这岂不是每次看路径都需要<kbd>pwd</kbd>吗?烦不烦啊!”别急,问题很好修改。</p> | ||
88 | <pre><code>cd ~ | ||
89 | vim .bashrc | ||
90 | # 找到PS1=,将其修改为PS1="\u@\h:\w\$" | ||
91 | |||
92 | # 加载.bashrc | ||
93 | source .bashrc</code></pre> | ||
94 | <p>改好之后也许还是没有变化,我就是这样的。后来在处理其它问题的时候,我发现了根本原因所在:我 TM 创建用户的时候忘了加<kbd>-s</kbd>参数,导致新用户的默认 shell | ||
95 | 是<kbd>sh</kbd>而不是<kbd>bash</kbd>。所以,我们需要修改一下新用户的默认 shell。</p> | ||
96 | <pre><code># 查看当前用户的默认shell | ||
97 | echo $SHELL | ||
98 | |||
99 | # 修改默认shell | ||
100 | sudo chsh -s /bin/bash player</code></pre> | ||
101 | <p>这时再重新加载,就会发现,一切都正常了,一切是那么美好。</p> | ||
102 | <p>除此之外,我们也许想要调整命令行显示的用户名、路径之类内容的颜色,会想让命令行像 git bash 一样当我们进入 git 仓库的时候显示当前在什么分支。针对这两项需求,我修改了一下我的.bashrc | ||
103 | 文件,现在其内容如下:</p> | ||
104 | <pre><code># ~/.bashrc: executed by bash(1) for non-login shells. | ||
105 | # see /usr/share/doc/bash/examples/startup-files (in the package bash-doc) | ||
106 | # for examples | ||
107 | |||
108 | # If not running interactively, don't do anything | ||
109 | case $- in | ||
110 | *i*) ;; | ||
111 | *) return;; | ||
112 | esac | ||
113 | |||
114 | # don't put duplicate lines or lines starting with space in the history. | ||
115 | # See bash(1) for more options | ||
116 | HISTCONTROL=ignoreboth | ||
117 | |||
118 | # append to the history file, don't overwrite it | ||
119 | shopt -s histappend | ||
120 | |||
121 | # for setting history length see HISTSIZE and HISTFILESIZE in bash(1) | ||
122 | HISTSIZE=1000 | ||
123 | HISTFILESIZE=2000 | ||
124 | |||
125 | # check the window size after each command and, if necessary, | ||
126 | # update the values of LINES and COLUMNS. | ||
127 | shopt -s checkwinsize | ||
128 | |||
129 | # If set, the pattern "**" used in a pathname expansion context will | ||
130 | # match all files and zero or more directories and subdirectories. | ||
131 | #shopt -s globstar | ||
132 | |||
133 | # make less more friendly for non-text input files, see lesspipe(1) | ||
134 | [ -x /usr/bin/lesspipe ] && eval "$(SHELL=/bin/sh lesspipe)" | ||
135 | |||
136 | # set variable identifying the chroot you work in (used in the prompt below) | ||
137 | if [ -z "${debian_chroot:-}" ] && [ -r /etc/debian_chroot ]; then | ||
138 | debian_chroot=$(cat /etc/debian_chroot) | ||
139 | fi | ||
140 | |||
141 | # set a fancy prompt (non-color, unless we know we "want" color) | ||
142 | case "$TERM" in | ||
143 | xterm-color|*-256color) color_prompt=yes;; | ||
144 | esac | ||
145 | |||
146 | # uncomment for a colored prompt, if the terminal has the capability; turned | ||
147 | # off by default to not distract the user: the focus in a terminal window | ||
148 | # should be on the output of commands, not on the prompt | ||
149 | #force_color_prompt=yes | ||
150 | |||
151 | if [ -n "$force_color_prompt" ]; then | ||
152 | if [ -x /usr/bin/tput ] && tput setaf 1 >&/dev/null; then | ||
153 | # We have color support; assume it's compliant with Ecma-48 | ||
154 | # (ISO/IEC-6429). (Lack of such support is extremely rare, and such | ||
155 | # a case would tend to support setf rather than setaf.) | ||
156 | color_prompt=yes | ||
157 | else | ||
158 | color_prompt= | ||
159 | fi | ||
160 | fi | ||
161 | |||
162 | # 简单修改一下颜色 | ||
163 | if [ "$color_prompt" = yes ]; then | ||
164 | PS1='${debian_chroot:+($debian_chroot)}\[\033[01;32m\]\u@\h\[\033[00m\]:\[\033[01;34m\]\[\033[00m\]' | ||
165 | else | ||
166 | PS1='${debian_chroot:+($debian_chroot)}\u@\h:' | ||
167 | fi | ||
168 | |||
169 | # 如果是git仓库,在命令行提示符之前用不同的颜色显示当前分支 | ||
170 | PS1="$PS1"'\[\033[33m\]' # change to yellow | ||
171 | PS1="$PS1"'\w' # pwd | ||
172 | PS1="$PS1"'\[\033[36m\]' # change color to cyan | ||
173 | PS1="$PS1"'`__git_ps1`' # git branch | ||
174 | PS1="$PS1"'\[\033[0m\]' # change color | ||
175 | #PS1="$PS1"'\n' # new line | ||
176 | PS1="$PS1"'\$ ' # prompt: always $ | ||
177 | unset color_prompt force_color_prompt | ||
178 | |||
179 | # If this is an xterm set the title to user@host:dir | ||
180 | case "$TERM" in | ||
181 | xterm*|rxvt*) | ||
182 | PS1="\[\e]0;${debian_chroot:+($debian_chroot)}\u@\h: \w\a\]$PS1" | ||
183 | ;; | ||
184 | *) | ||
185 | ;; | ||
186 | esac | ||
187 | |||
188 | # enable color support of ls and also add handy aliases | ||
189 | if [ -x /usr/bin/dircolors ]; then | ||
190 | test -r ~/.dircolors && eval "$(dircolors -b ~/.dircolors)" || eval "$(dircolors -b)" | ||
191 | alias ls='ls --color=auto' | ||
192 | #alias dir='dir --color=auto' | ||
193 | #alias vdir='vdir --color=auto' | ||
194 | |||
195 | alias grep='grep --color=auto' | ||
196 | alias fgrep='fgrep --color=auto' | ||
197 | alias egrep='egrep --color=auto' | ||
198 | fi | ||
199 | |||
200 | # colored GCC warnings and errors | ||
201 | #export GCC_COLORS='error=01;31:warning=01;35:note=01;36:caret=01;32:locus=01:quote=01' | ||
202 | |||
203 | # some more ls aliases | ||
204 | alias ll='ls -alF' | ||
205 | alias la='ls -A' | ||
206 | alias l='ls -CF' | ||
207 | |||
208 | # Add an "alert" alias for long running commands. Use like so: | ||
209 | # sleep 10; alert | ||
210 | alias alert='notify-send --urgency=low -i "$([ $? = 0 ] && echo terminal || echo error)" "$(history|tail -n1|sed -e '\''s/^\s*[0-9]\+\s*//;s/[;&|]\s*alert$//'\'')"' | ||
211 | |||
212 | # Alias definitions. | ||
213 | # You may want to put all your additions into a separate file like | ||
214 | # ~/.bash_aliases, instead of adding them here directly. | ||
215 | # See /usr/share/doc/bash-doc/examples in the bash-doc package. | ||
216 | |||
217 | if [ -f ~/.bash_aliases ]; then | ||
218 | . ~/.bash_aliases | ||
219 | fi | ||
220 | |||
221 | # enable programmable completion features (you don't need to enable | ||
222 | # this, if it's already enabled in /etc/bash.bashrc and /etc/profile | ||
223 | # sources /etc/bash.bashrc). | ||
224 | if ! shopt -oq posix; then | ||
225 | if [ -f /usr/share/bash-completion/bash_completion ]; then | ||
226 | . /usr/share/bash-completion/bash_completion | ||
227 | elif [ -f /etc/bash_completion ]; then | ||
228 | . /etc/bash_completion | ||
229 | fi | ||
230 | fi | ||
231 | echo This\ is\ .bashrc. | ||
232 | |||
233 | alias pip='python -m pip' | ||
234 | # 设置bash使用vim模式,允许使用0、dw之类常用vim快捷键,方便命令编辑 | ||
235 | set -o vi | ||
236 | # 有时候我们需要一些魔法 | ||
237 | export http_proxy=http://127.0.0.1:7890 | ||
238 | export https_proxy=http://127.0.0.1:7890</code></pre> | ||
239 | <h1 id="配置-ssh">配置 ssh</h1> | ||
102 | <p>用着用着,就会发现:我为啥一直在浏览器里搞命令行?要是能直接在本地的命令行里搞就好了。当然,阿里云能满足我们。</p> | 240 | <p>用着用着,就会发现:我为啥一直在浏览器里搞命令行?要是能直接在本地的命令行里搞就好了。当然,阿里云能满足我们。</p> |
103 | <h3 id="ssh-简介">ssh 简介</h3> | 241 | <h2 id="ssh-简介">ssh 简介</h2> |
104 | <p>什么是 ssh?ssh 是 Secure Shell 的缩写,中文名为安全外壳协议。ssh 是一种加密的网络协议,可以在不安全的网络中为网络服务提供安全的传输环境。ssh | 242 | <p>什么是 ssh?ssh 是 Secure Shell 的缩写,中文名为安全外壳协议。ssh 是一种加密的网络协议,可以在不安全的网络中为网络服务提供安全的传输环境。ssh |
105 | 是目前较可靠,专为远程登录会话和其他网络服务提供安全性的协议。利用 ssh 协议可以有效防止远程管理过程中的信息泄露问题。通过 ssh 可以对所有传输的数据进行加密,也能够防止 DNS 欺骗和 IP | 243 | 是目前较可靠,专为远程登录会话和其他网络服务提供安全性的协议。利用 ssh 协议可以有效防止远程管理过程中的信息泄露问题。通过 ssh 可以对所有传输的数据进行加密,也能够防止 DNS 欺骗和 IP |
106 | 欺骗。ssh | 244 | 欺骗。ssh 还支持在两个远程主机之间进行数据的传输。</p> |
107 | 还支持在两个远程主机之间进行数据的传输。</p> | ||
108 | <p>说了也记不住,直接开干!</p> | 245 | <p>说了也记不住,直接开干!</p> |
109 | <h3 id="配置安全组规则">配置安全组规则</h3> | 246 | <h2 id="配置安全组规则">配置安全组规则</h2> |
110 | <p>在阿里云控制台左上角点击,选择<code>云服务器ECS</code>,会进入云服务器实例控制界面。在左侧菜单栏寻找<code>安全组</code> ,添加安全组规则。按要求添加新的安全组规则,开放 TCP | 247 | <p>在阿里云控制台左上角点击,选择<code>云服务器ECS</code>,会进入云服务器实例控制界面。在左侧菜单栏寻找<code>安全组</code> ,添加安全组规则。按要求添加新的安全组规则,开放 TCP |
111 | 协议对应的 | 248 | 协议对应的 22 端口。(其实可以都开了,但是毕竟不安全,建议后续按需开放。)</p> |
112 | 22 端口。(其实可以都开了,但是毕竟不安全,建议后续按需开放。)</p> | ||
113 | <p>接着,在菜单栏中点击<code>密钥对</code>,创建新的密钥对。<strong>完成之后会自动下载一个.pem 文件,我们需要将该文件保存好,丢失之后无法找回,这是 ssh 连接的密钥。</strong> | 249 | <p>接着,在菜单栏中点击<code>密钥对</code>,创建新的密钥对。<strong>完成之后会自动下载一个.pem 文件,我们需要将该文件保存好,丢失之后无法找回,这是 ssh 连接的密钥。</strong> |
114 | </p> | 250 | </p> |
115 | <p>为了方便用别名连接,我们需要一个配置文件:</p> | 251 | <p>为了方便用别名连接,我们需要一个配置文件:</p> |
@@ -117,7 +253,7 @@ sudo apt install make cmake | |||
117 | <li>将上述.pem 文件移动到家目录<code>C:\用户\当前用户名</code>的<kbd>.ssh</kbd>文件夹下</li> | 253 | <li>将上述.pem 文件移动到家目录<code>C:\用户\当前用户名</code>的<kbd>.ssh</kbd>文件夹下</li> |
118 | <li>在<kbd>.ssh</kbd>文件夹下创建文件<kbd>config</kbd>,内容如下:</li> | 254 | <li>在<kbd>.ssh</kbd>文件夹下创建文件<kbd>config</kbd>,内容如下:</li> |
119 | </ul> | 255 | </ul> |
120 | <pre><code class="language-plaintext"># 给服务器起的别名,这里是aliyun | 256 | <pre><code># 给服务器起的别名,这里是aliyun |
121 | Host aliyun | 257 | Host aliyun |
122 | # IP地址,这里以127.0.0.1代替,但真实操作需要换成服务器的公网ip,在控制台就能看到 | 258 | # IP地址,这里以127.0.0.1代替,但真实操作需要换成服务器的公网ip,在控制台就能看到 |
123 | IP 127.0.0.1 | 259 | IP 127.0.0.1 |
@@ -132,208 +268,21 @@ User player | |||
132 | ssh aliyun</code></pre> | 268 | ssh aliyun</code></pre> |
133 | <h1 id="拥抱-vim">拥抱 vim</h1> | 269 | <h1 id="拥抱-vim">拥抱 vim</h1> |
134 | <p>既然我们选择了在命令行里使用,编辑文件当然是比较常用 vim 了。当然 neovim 与其他命令行编辑器也很好,但我不熟悉,请参照网上其他教程进行配置。</p> | 270 | <p>既然我们选择了在命令行里使用,编辑文件当然是比较常用 vim 了。当然 neovim 与其他命令行编辑器也很好,但我不熟悉,请参照网上其他教程进行配置。</p> |
135 | <p>我的 vim 学习与配置<a href="https://www.qin-juan-he-zhu.top/code/linux/vim.html">链接</a>中有如下一些基本配置:</p> | 271 | <p>我的 vim 学习与配置<a href="https://www.qin-juan-ge-zhu.top/code/linux/vim.html">链接</a>中有如下一些基本配置:</p> |
136 | <ul> | 272 | <ul> |
137 | <li>同时开启绝对行号与相对行号,则当前行显示真实行号,其他行显示相对于当前行为上/下第几行,方便跳转。</li> | ||
138 | <li>代码中常常需要的<code>()[]{}</code>自动匹配</li> | ||
139 | <li>自动缩进和智能缩进,但似乎没啥鸟用</li> | ||
140 | <li>vim 自带的代码高亮,感觉还是相当不错的</li> | 273 | <li>vim 自带的代码高亮,感觉还是相当不错的</li> |
274 | <li>代码中常常需要的<code>()[]{}</code>自动匹配</li> | ||
275 | <li>同时开启绝对行号与相对行号,则当前行显示真实行号,其他行显示相对于当前行为上/下第几行,方便跳转。</li> | ||
276 | <li>代码按语法缩进,并且每次保存或退出时自动格式化</li> | ||
141 | <li>当前行/当前列各给一个颜色,因为有的平台光标显示很不明显,正常插入模式一般也不好确定光标位置,这样方便知道光标在哪。</li> | 277 | <li>当前行/当前列各给一个颜色,因为有的平台光标显示很不明显,正常插入模式一般也不好确定光标位置,这样方便知道光标在哪。</li> |
142 | <li>自动折行,即当前行太长了会被折回来显示,但还是同一行,方便一次性看完整。</li> | 278 | <li>自动折行,即当前行太长了会被折回来显示,但还是同一行,方便一次性看完整。</li> |
143 | <li>一些快捷键,如<kbd>ta</kbd>到行末(替代够不到的<kbd>$</kbd>),<kbd>end</kbd>来到文件末(替代难记的<kbd>G</kbd>),还有<kbd>up/ne</kbd>上/下移动 | 279 | <li>一些快捷键,如<kbd>L</kbd>到行末(替代够不到的<kbd>$</kbd>),还有<kbd>K/J</kbd>上/下移动 10 行等。</li> |
144 | 20 行等。</li> | 280 | <li>基本的状态栏设置,包括文件名、git 分支、文件编码、文件类型、总行数等信息,方便查看。</li> |
145 | <li>自动识别文件类型,并对<code>*.html, *.c, *.cpp, *.python</code>等文件在保存<kbd>:w<Enter> | 281 | <li>自动识别文件类型,并对<code>*.html, *.c, *.cpp, *.python</code>等文件在保存<kbd>:w<Enter></kbd>时自动格式化</li> |
146 | </kbd>时自动格式化,在正常或选择模式下<kbd>Ctrl+p</kbd>一键注释(别骂了别骂了,<kbd>Ctrl+/</kbd>我属实是设置不出来)</li> | 282 | <li>插件和其他重重设置</li> |
147 | </ul> | 283 | </ul> |
148 | <h2 id="插件管理">插件管理</h2> | ||
149 | <p>这里使用了几个插件,插件管理目前用的是评价一般的<code>Vundle</code>。个人评价,下载插件啥的属实是慢,每次都报错报到怀疑人生,害得我都得手动安装然后在里边添加,但是别的目前还不会使。气死偶勒! | 284 | <p>这里使用了几个插件,插件管理目前用的是评价一般的<code>Vundle</code>。个人评价,下载插件啥的属实是慢,每次都报错报到怀疑人生,害得我都得手动安装然后在里边添加,但是别的目前还不会使。气死偶勒! |
150 | </p> | 285 | </p> |
151 | <p>这里需要注意的是,上文提到我喜欢把有用的东西放在<code>~/useful</code>下,但是如果直接这样写路径,其他用户如 root | ||
152 | 等使用时候就会报错,人家的主目录第下没这个东西!所以,在写路径的位置,切记要写绝对路径!!!</p> | ||
153 | <h1 id="游戏">游戏</h1> | ||
154 | <blockquote> | ||
155 | <p>All work no play makes Jack a doll boy.</p> | ||
156 | </blockquote> | ||
157 | <p>经过了上边的配置,写代码的控制基本完善了,但是服务器怎么能只拿来干活呢?没有游戏怎么行?我们要快乐!</p> | ||
158 | <p>目前我的服务器上装了三个游戏</p> | ||
159 | <ul> | ||
160 | <li>俄罗斯方块,装的是<code>tint</code></li> | ||
161 | <li>扫雷,装的是<code>miebash</code></li> | ||
162 | <li>贪吃蛇,找不到力(允悲</li> | ||
163 | </ul> | ||
164 | <p>安装方法网上都有,就不多说了。玩得愉快!</p> | ||
165 | <h1 id="git-远程存储">git 远程存储</h1> | ||
166 | <p>君不闻,《西游记》有云:</p> | ||
167 | <blockquote> | ||
168 | <p>争名夺利几时休,早起迟眠不自由。</p> | ||
169 | <p>骑着驴骡思骏马,官居宰相望王侯。</p> | ||
170 | </blockquote> | ||
171 | <p>完成了代码环境的配置,我就在想,要是有一个自己的远程 git 存储库该多好?</p> | ||
172 | <p>说干就干,我找到了 gitlab 安装教程,信心满满,谁知“先帝创业未半而中道崩殂”。</p> | ||
173 | <h2 id="gitlab">gitlab?</h2> | ||
174 | <p>我按照网上的教程,更新了清华镜像源,下载安装 gitlab。孰料一次一次在安装过程卡死,而且情况十分严重,CPU 占用率长期维持在 50%以上,内存更是飙升到 | ||
175 | 90%,现在不仅安装进行不下去了,甚至一切操作都不认识,包括<kbd>Ctrl+C</kbd>想要杀死当前进程的请求也会石沉大海。没办法,只得强制重启。</p> | ||
176 | <p>如是反复若干次,最终死心,将 gitlab 彻底从云服务器卸载,“革职为民,永不叙用”。</p> | ||
177 | <p>经过上网查找,原因果然出现在配置上。<strong>gitlab 推荐的最小内存是 4G,但是目前我们只有 | ||
178 | 2G,内存爆满也就理所当然了。</strong>没办法,期限之内不能更改配置,就算能更改,我都穷到薅资本主义羊毛了,还能有钱升级配置?笑死。</p> | ||
179 | <h2 id="就这">就这?</h2> | ||
180 | <p>就因为这个问题,我们就不干了?这是不行滴,小同志。</p> | ||
181 | <p>不久之后,黄四郎同志发来了一篇博客:<a href="https://www.ruanyifeng.com/blog/2022/10/git-server.html">最简单的 git 服务器</a></p> | ||
182 | <p>看起来似乎不错,但是失之简略。幸好,我在 git 官方教程<a href="https://git-scm.com/book/zh/v2">Pro | ||
183 | Git</a>上找到了另一部分,两个拼一拼、试一试,最终成功了。 | ||
184 | </p> | ||
185 | <p>注意,本地和远程都需要安装 git,相信能看这里的读者应该已经是安装过了,这里不再赘述。</p> | ||
186 | <h3 id="前置">前置</h3> | ||
187 | <p>在创建存储库之前,我想我们应该做些什么。</p> | ||
188 | <p>对了,创建新用户吧!按照上述写的步骤创建一个新用户并为之设置密码,可以不用加入 sudo 组,因为我们不需要这个用户来操作服务器,只需要用来存储代码就行了。</p> | ||
189 | <h3 id="本地存储库">本地存储库</h3> | ||
190 | <p>本地我们需要一个存储库,如果已有的则可以忽略本部分。</p> | ||
191 | <pre><code class="language-bash"># 创建本地存储库,这里叫test吧 | ||
192 | mkdir test | ||
193 | cd test | ||
194 | git init</code></pre> | ||
195 | <p>但是本方法要求本地存储库必须已经有 commit(s)。好办,就写一个常用的<code>.gitignore</code>好了。</p> | ||
196 | <pre><code class="language-plaintext">*.sh | ||
197 | *.bat | ||
198 | *.exe | ||
199 | *.dll | ||
200 | *.so | ||
201 | *.[oa] | ||
202 | *.idea</code></pre> | ||
203 | <p>而后,为了方便提交操作,可以再创建一个<code>push.sh</code>脚本:</p> | ||
204 | <pre><code class="language-bash">#!/bin/bash | ||
205 | git add . | ||
206 | git commit | ||
207 | git push</code></pre> | ||
208 | <p>注意,创建<code>push.sh</code>一定要在<code>.gitignore</code>之后,否则不会被忽略的。</p> | ||
209 | <h3 id="远程仓库">远程仓库</h3> | ||
210 | <p>就一句话:</p> | ||
211 | <pre><code class="language-bash"># 创建远程仓库,这里也叫test吧 | ||
212 | ssh git@127.0.0.1 git init --bare test.git</code></pre> | ||
213 | <p>注意,本处指明的 git 为远程用户名,127.0.0.1 代表云服务器的公网 ip,test.git 为远程仓库名。使用的时候都需要换成自己的。</p> | ||
214 | <h3 id="本地与远程连接">本地与远程连接</h3> | ||
215 | <p>本地和远程都有了,下一步就是建立联系了。</p> | ||
216 | <pre><code class="language-bash"># 本地添加远程 | ||
217 | git remote add origin git@127.0.0.1:test.git | ||
218 | |||
219 | # 此时尚不能直接推送,因为并未指定上游对应分支,需要指定 | ||
220 | git push --set-upstream origin master | ||
221 | |||
222 | # 之后就可以直接推送了 | ||
223 | sh push.sh</code></pre> | ||
224 | <h3 id="自动化">自动化</h3> | ||
225 | <p>怎么样,操作看起来又臭又长吧?我也这么觉得。所以我写了两份代码,供同时自动创建本地仓库和远程对应仓库并建立所有对应关系之用。</p> | ||
226 | <pre><code class="language-c">/* | ||
227 | * gitadd.c | ||
228 | */ | ||
229 | #include <stdio.h> | ||
230 | #include <stdlib.h> | ||
231 | #include <string.h> | ||
232 | #include <direct.h> | ||
233 | |||
234 | int main() | ||
235 | { | ||
236 | char local_name[100],remote_name[100],tmp[100]; | ||
237 | printf("您好!\n请输入本地仓库名:"); | ||
238 | gets(local_name); | ||
239 | printf("请输入远程仓库名:"); | ||
240 | gets(remote_name); | ||
241 | |||
242 | //远程仓库创建 | ||
243 | strcpy(tmp,"ssh aliyun-git git init --bare "); | ||
244 | strcat(remote_name,".git"); | ||
245 | strcat(tmp,remote_name); | ||
246 | system(tmp); | ||
247 | |||
248 | //本地仓库创建 | ||
249 | mkdir(local_name); | ||
250 | chdir(local_name); | ||
251 | system("git init"); | ||
252 | |||
253 | //本地仓库初始化 | ||
254 | FILE* file; | ||
255 | //编写.gitignore | ||
256 | file=fopen(".gitignore","w"); | ||
257 | fprintf(file,"*.sh\n"); | ||
258 | fprintf(file,"*.bat\n"); | ||
259 | fprintf(file,"*.exe\n"); | ||
260 | fprintf(file,"*.[oa]\n"); | ||
261 | fprintf(file,"*.pyc\n"); | ||
262 | fprintf(file,"__pycache__\n"); | ||
263 | fprintf(file,"*.vscode\n"); | ||
264 | fprintf(file,"*.swp\n"); | ||
265 | fclose(file); | ||
266 | //编写push.sh | ||
267 | file=fopen("push.sh","w"); | ||
268 | fprintf(file,"git add .\n"); | ||
269 | fprintf(file,"git commit\n"); | ||
270 | fprintf(file,"git push"); | ||
271 | fclose(file); | ||
272 | |||
273 | //提交初始化commit | ||
274 | system("git add ."); | ||
275 | system("git commit -m \"Initial commit\""); | ||
276 | strcpy(tmp,"git remote add origin aliyun-git:"); | ||
277 | strcat(tmp,remote_name); | ||
278 | system(tmp); | ||
279 | system("git push --set-upstream origin master"); | ||
280 | system("git push"); | ||
281 | |||
282 | printf("完成!"); | ||
283 | return 0; | ||
284 | }</code></pre> | ||
285 | <pre><code class="language-python"># gitadd.py | ||
286 | import os | ||
287 | |||
288 | if __name__ == '__main__': | ||
289 | print("你好!") | ||
290 | print("请输入本地仓库名:") | ||
291 | local_name = input() | ||
292 | print("请输入远程仓库名:") | ||
293 | remote_name = input() | ||
294 | os.system("ssh aliyun-git git init --bare "+remote_name+".git") | ||
295 | |||
296 | # 创建本地仓库 | ||
297 | os.system("mkdir "+local_name) | ||
298 | os.chdir(local_name) | ||
299 | print("本地仓库已建立!") | ||
300 | print("当前路径为"+os.getcwd()) | ||
301 | |||
302 | # 初始化git仓库 | ||
303 | os.system("git init") | ||
304 | |||
305 | # 以utf8编码编写.gitignore文件 | ||
306 | with open(".gitignore", "w", encoding="utf8") as f: | ||
307 | f.write("*.pyc\n") | ||
308 | f.write("__pycache__/\n") | ||
309 | f.write("*.exe\n") | ||
310 | f.write("*.[oa]\n") | ||
311 | f.write("*.swp\n") | ||
312 | f.write("*.sh\n") | ||
313 | f.write("*.bat\n") | ||
314 | |||
315 | # 编写push.sh文件 | ||
316 | with open("push.sh", "w", encoding="utf8") as f: | ||
317 | f.write("git add .\n") | ||
318 | f.write("git commit\n") | ||
319 | f.write("git push\n") | ||
320 | |||
321 | print("本地仓库初始化完成!") | ||
322 | |||
323 | # 连接远程仓库 | ||
324 | os.system("git add .") | ||
325 | os.system("git commit -m \"Initial commit\"") | ||
326 | os.system("git remote add origin aliyun-git:"+remote_name+".git") | ||
327 | os.system("git push --set-upstream origin master") | ||
328 | os.system("git push") | ||
329 | |||
330 | print("完成!")</code></pre> | ||
331 | <p><strong>Attention please:</strong> | ||
332 | 本地仓库名和远程仓库名不要加后缀,程序会自动添加。另外,如果要修改,在转换本地当前工作目录的时候,<strong>一定要使用语言自带的调整当前工作目录的函数,不要调用系统命令</strong>,否则会出现编译器/解释器自动将程序作为多线程执行,导致一个线程进去了然后线程当场去世,后续的操作在另一个线程,工作目录根本没发生变化,也就是把原来所在的位置给 | ||
333 | git 初始化了,出现奇奇怪怪的错误。</p> | ||
334 | <h3 id="其他">其他</h3> | ||
335 | <p>需要明白的是,上述方法在远端创立的<strong>只是一个裸仓库</strong>(即只是我们本地仓库的<code>.git</code>文件夹),不包含工作目录,所以不能直接在远端进行操作,需要在本地进行操作,然后推送到远端。但是<code>git clone</code>的时候,本地得到的是一个完整的仓库,而其他使用也与普通仓库无异。 | ||
336 | </p> | ||
337 | <h1 id="来看看在线-vscode-吧">来看看在线 VSCode 吧!</h1> | 286 | <h1 id="来看看在线-vscode-吧">来看看在线 VSCode 吧!</h1> |
338 | <p>本来,一切就这样完美收官、从胜利走向更大的胜利了。坏就坏在我按照五个月前的计划添置了一个 | 287 | <p>本来,一切就这样完美收官、从胜利走向更大的胜利了。坏就坏在我按照五个月前的计划添置了一个 |
339 | iPad。在最初的设想里它就不单单是当平板做笔记看书用的,而是要充当电脑的编程功能,来在必要的时刻随拿随走,替代我这十几斤重、续航还短的游戏本的。</p> | 288 | iPad。在最初的设想里它就不单单是当平板做笔记看书用的,而是要充当电脑的编程功能,来在必要的时刻随拿随走,替代我这十几斤重、续航还短的游戏本的。</p> |
@@ -347,9 +296,9 @@ if __name__ == '__main__': | |||
347 | <p>但是我这里 curl 不知何故一直不能用,wget 也不好使,只能在 gayhub 仓库的<a | 296 | <p>但是我这里 curl 不知何故一直不能用,wget 也不好使,只能在 gayhub 仓库的<a |
348 | href="https://github.com/coder/code-server/releases">Releases</a>里找到最新版适合的安装包手动安装了。我是 | 297 | href="https://github.com/coder/code-server/releases">Releases</a>里找到最新版适合的安装包手动安装了。我是 |
349 | Ubuntu22.04LTS,系统架构 amd64,所以下载了对应的.deb 安装包。</p> | 298 | Ubuntu22.04LTS,系统架构 amd64,所以下载了对应的.deb 安装包。</p> |
350 | <pre><code class="language-bash"># 本地命令行中运行 | 299 | <pre><code># 本地命令行中运行 |
351 | scp code-server-4.13.0-linux-amd64.deb aliyun:code-server-4.13.0-linux-amd64.deb</code></pre> | 300 | scp code-server-4.13.0-linux-amd64.deb aliyun:code-server-4.13.0-linux-amd64.deb</code></pre> |
352 | <pre><code class="language-bash"># 云服务器命令行中运行 | 301 | <pre><code># 云服务器命令行中运行 |
353 | 302 | ||
354 | # 安装 | 303 | # 安装 |
355 | sudo dpkg -i code-server-4.13.0-linux-amd64.deb</code></pre> | 304 | sudo dpkg -i code-server-4.13.0-linux-amd64.deb</code></pre> |
@@ -358,14 +307,11 @@ sudo dpkg -i code-server-4.13.0-linux-amd64.deb</code></pre> | |||
358 | <p>安装完成后,就可以直接运行了。</p> | 307 | <p>安装完成后,就可以直接运行了。</p> |
359 | <pre><code>code-server</code></pre> | 308 | <pre><code>code-server</code></pre> |
360 | <p>就可以直接运行了,但是这样只能在本地浏览器运行,而我们需要的是远程运行。算了,直接上脚本!</p> | 309 | <p>就可以直接运行了,但是这样只能在本地浏览器运行,而我们需要的是远程运行。算了,直接上脚本!</p> |
361 | <pre><code class="language-shell">#!/bin/sh | 310 | <pre><code>#!/bin/sh |
362 | code-server --host "0.0.0.0"> /home/player/useful/codeserver-out.txt 2>&1 & | 311 | code-server --host "0.0.0.0"> /home/player/useful/codeserver-out.txt 2>&1 &</code></pre> |
363 | echo 启动完成!程序已在后台运行。 | ||
364 | echo 如需杀死进程,请以jobs查看后台程序编号并kill %num | ||
365 | echo 或ps查看进程号并kill之。</code></pre> | ||
366 | <p>其中第一条指令是设置允许外界访问,并指定了由后台运行、所有输出写进<code>/home/player/useful/codeserver.txt</code>文件中。后续均为输出信息。</p> | 312 | <p>其中第一条指令是设置允许外界访问,并指定了由后台运行、所有输出写进<code>/home/player/useful/codeserver.txt</code>文件中。后续均为输出信息。</p> |
367 | <p>初次这样设置,直接打开会有密码,密码的存储文件在<code>~/.config/code-server/config.yaml</code>中,可以直接修改,也可以使用命令行修改。比如我当前的设置为:</p> | 313 | <p>初次这样设置,直接打开会有密码,密码的存储文件在<code>~/.config/code-server/config.yaml</code>中,可以直接修改,也可以使用命令行修改。比如我当前的设置为:</p> |
368 | <pre><code class="language-yaml">bind-addr: 127.0.0.1:8080 | 314 | <pre><code>bind-addr: 127.0.0.1:8080 |
369 | auth: password | 315 | auth: password |
370 | password: player | 316 | password: player |
371 | cert: false</code></pre> | 317 | cert: false</code></pre> |
@@ -374,10 +320,7 @@ cert: false</code></pre> | |||
374 | <h2 id="杀死">杀死</h2> | 320 | <h2 id="杀死">杀死</h2> |
375 | <p>设置好之后会需要重新启动程序,我一般选择杀死然后重新运行启动脚本。</p> | 321 | <p>设置好之后会需要重新启动程序,我一般选择杀死然后重新运行启动脚本。</p> |
376 | <p>杀死的方法:</p> | 322 | <p>杀死的方法:</p> |
377 | <pre><code class="language-bash"># 运行指令 | 323 | <pre><code>$ ps -ef | grep code-server |
378 | ps -ef | grep code-server | ||
379 | |||
380 | # 输出结果如下 | ||
381 | player 2621 1 0 May29 ? 00:00:00 /usr/lib/code-server/lib/node /usr/lib/code-server --host 0.0.0.0 | 324 | player 2621 1 0 May29 ? 00:00:00 /usr/lib/code-server/lib/node /usr/lib/code-server --host 0.0.0.0 |
382 | player 2646 2621 0 May29 ? 00:03:31 /usr/lib/code-server/lib/node /usr/lib/code-server/out/node/entry | 325 | player 2646 2621 0 May29 ? 00:03:31 /usr/lib/code-server/lib/node /usr/lib/code-server/out/node/entry |
383 | player 3079 2646 0 May29 ? 00:00:57 /usr/lib/code-server/lib/node /usr/lib/code-server/lib/vscode/out/bootstrap-fork --type=ptyHost --logsPath /home/player/.local/share/code-server/logs/20230529T211752 | 326 | player 3079 2646 0 May29 ? 00:00:57 /usr/lib/code-server/lib/node /usr/lib/code-server/lib/vscode/out/bootstrap-fork --type=ptyHost --logsPath /home/player/.local/share/code-server/logs/20230529T211752 |
@@ -390,8 +333,8 @@ player 465057 464494 0 13:42 ? 00:00:35 /usr/lib/code-server/lib/nod | |||
390 | player 579173 574648 0 16:44 pts/1 00:00:00 grep --color=auto code-server | 333 | player 579173 574648 0 16:44 pts/1 00:00:00 grep --color=auto code-server |
391 | 334 | ||
392 | # 只需要杀死根进程也就是第一个进程就好,下边的数据需要改成当前对应进程号 | 335 | # 只需要杀死根进程也就是第一个进程就好,下边的数据需要改成当前对应进程号 |
393 | kill -9 2621</code></pre> | 336 | $ kill -9 2621</code></pre> |
394 | <p>重新启动之后,就可以用<code><服务器公网ip>:<指定端口></code>访问在线 VSCode 了,初次访问会比较慢,耐心等待吧。</p> | 337 | <p>重新启动之后,就可以用<code>ip:post</code>访问在线 VSCode 了,初次访问会比较慢,耐心等待吧。</p> |
395 | <h2 id="同步设置">同步设置</h2> | 338 | <h2 id="同步设置">同步设置</h2> |
396 | <p>在线 VSCode 设置好了,但是为了进一步抹平在线端与本地端的差异,最好做一下同步设置处理。网上大多数教程给的都是利用插件<a | 339 | <p>在线 VSCode 设置好了,但是为了进一步抹平在线端与本地端的差异,最好做一下同步设置处理。网上大多数教程给的都是利用插件<a |
397 | href="https://marketplace.visualstudio.com/items?itemName=Shan.code-settings-sync">Settings | 340 | href="https://marketplace.visualstudio.com/items?itemName=Shan.code-settings-sync">Settings |
@@ -425,50 +368,56 @@ kill -9 2621</code></pre> | |||
425 | github 上,一旦把一级域名解析到服务器,就要出问题。</p> | 368 | github 上,一旦把一级域名解析到服务器,就要出问题。</p> |
426 | <p>除此之外,记得到阿里云 ecs 那里,在安全组规则中,开放 http/https 对应的 80/443 端口!两个都要开!</p> | 369 | <p>除此之外,记得到阿里云 ecs 那里,在安全组规则中,开放 http/https 对应的 80/443 端口!两个都要开!</p> |
427 | <h2 id="icp-备案">ICP 备案</h2> | 370 | <h2 id="icp-备案">ICP 备案</h2> |
428 | <p>众所周知,我国是工人阶级领导的、以工农联盟为基础的、人民民主专政的社会主义国家,一切事关宣传、舆论的东西都要在党的领导下,都要经过审查备案,包括建网站和域名 https 使用。</p> | 371 | <p>众所周知,我国是“工人阶级领导的、以工农联盟为基础的、人民民主专政的社会主义”国家,一切事关宣传、舆论的东西都要在党的领导下,都要经过审查备案,包括建网站和域名 https 使用。ICP |
429 | <p>ICP 备案的原则是:用谁的服务器,在谁那里备案。即以我而言,我在腾讯云买的域名、做的 DNS 解析,但用的是阿里云的服务器,就要在阿里云做 ICP 备案。</p> | 372 | 备案的原则是:<strong>用谁的服务器,在谁那里备案</strong>。以我为例,我在腾讯云买的域名、做的 DNS 解析,但用的是阿里云的服务器,就要在阿里云做 ICP 备案。</p> |
430 | <p>备案流程按按着走就行,只是有几点需要特别注意:</p> | 373 | <p>备案流程按按着走就行,只是有几点需要特别注意:</p> |
431 | <ul> | 374 | <ul> |
432 | <li>备案的省份需要与户籍所在省一致</li> | 375 | <li>备案的省份需要与户籍所在省一致</li> |
433 | <li>备案网站名称不要太普通,容易和别人重复,会被客服小姐姐电话要求订正</li> | 376 | <li>备案网站名称不要太普通,容易和别人重复,会被客服小姐姐电话要求订正</li> |
434 | <li>网站名称不要出现“阁、楼、店”等一些容易误会为企业商业主体网站的名称,否则也会被退回。</li> | 377 | <li>网站名称不要出现“阁、楼、店”等一些容易误会为企业商业主体网站的名称,否则也会被退回。</li> |
435 | <li>网站用途描述要详细,尽量 30 字以上,绝对不要出现“博客”等词,审查非常非常严格,而且极大可能直接被 ban。即使你真要博客,不能暗度陈仓吗?</li> | 378 | <li>网站用途描述要详细,尽量 30 字以上 |
379 | <ul> | ||
380 | <li>用途描述绝对不要出现“博客”等词,审查非常非常严格,而且极大可能直接被 ban。即使你真要博客,不能暗度陈仓吗?</li> | ||
381 | <li>就算你要开评论区也不要说提及。<strong>朋友,你也不想你的博客隔三差五被警察同志拿着放大镜看吧?</strong></li> | ||
382 | </ul> | ||
383 | </li> | ||
436 | </ul> | 384 | </ul> |
437 | <p>提交备案之后。如有问题客服会打电话通知,并退回修改;阿里云初审需要大概不到 1 天,工信部短信确认需要 1 天,然后发送到备案省(即户籍所在省)的管局,时长最多 20 天,建议提前搞。我用了 6 天。</p> | 385 | <p>提交备案之后。如有问题客服会打电话通知,并退回修改;阿里云初审需要大概不到 1 天,工信部短信确认需要 1 天,然后发送到备案省(即户籍所在省)的管局,时长最多 20 天,建议提前搞。我用了 6 天。</p> |
438 | <h2 id="nginx-代理">Nginx 代理</h2> | 386 | <h2 id="nginx-代理">Nginx 代理</h2> |
439 | <p>都做好之后,访问<code>code.player.com</code>,发现啥也没有,浏览器说对面没反应。是的,因为 A 记录只解析到了 ip,并没有解析到指定端口,而 http 默认端口 80,https | 387 | <p>都做好之后,访问<code>code.player.com</code>,发现啥也没有,浏览器说对面没反应。是的,因为 A 记录只解析到了 ip,并没有解析到指定端口,而 http 默认端口 80,https |
440 | 默认端口 | 388 | 默认端口 443,都和上边 code-server 指定的端口 8080 不一样。</p> |
441 | 443,都和上边 code-server 指定的端口 8080 不一样。</p> | ||
442 | <p>没错,我们需要一个代理,根据访问的网址来对应到指定端口。</p> | 389 | <p>没错,我们需要一个代理,根据访问的网址来对应到指定端口。</p> |
443 | <h3 id="安装-1">安装</h3> | 390 | <h3 id="安装运行与常用操作">安装、运行与常用操作</h3> |
444 | <pre><code>sudo apt install nginx</code></pre> | 391 | <pre><code>sudo apt install nginx |
445 | <h3 id="运行-1">运行</h3> | 392 | # 安装完成后如果服务器重启,nginx会自动启动,手动启动命令如下 |
446 | <pre><code>nginx</code></pre> | 393 | sudo nginx |
394 | |||
395 | # 停止服务 | ||
396 | sudo nginx -s stop | ||
397 | |||
398 | # 重新加载配置文件(适用于修改了配置文件但不需要重启的场景) | ||
399 | sudo nginx -s reload</code></pre> | ||
447 | <h3 id="ssl-证书">SSL 证书</h3> | 400 | <h3 id="ssl-证书">SSL 证书</h3> |
448 | <p>想要使用 https 方式,还需要一个 SSL 证书,每个域名一个证。腾讯云比较坑,必须花钱。这里我用的是 Certbot 免费发放的证书。</p> | 401 | <p>想要使用 https 方式,还需要一个 SSL 证书,每个域名一个证。腾讯云比较坑,必须花钱。这里我用的是 Certbot 免费发放的证书。</p> |
449 | <p>首先,下载 Certbot:</p> | 402 | <p>首先,下载 Certbot:</p> |
450 | <pre><code>sudo apt install certbot</code></pre> | 403 | <pre><code>sudo apt install certbot</code></pre> |
451 | <p>在申请证书之前,需要停止 nginx 的工作,如果正在运行,按照上边说的方法杀死即可:</p> | 404 | <p>在申请证书之前,需要停止 nginx 的工作,如果正在运行,按照上边说的方法杀死,而后就可以申请证书辣!</p> |
452 | <pre><code class="language-bash">ps -ef | grep nginx | 405 | <pre><code># 申请证书 |
406 | sudo certbot certonly --standalone --email example@qq.com -d code.player.com | ||
453 | 407 | ||
454 | # 杀死显示的第一个进程 | 408 | # 除了使用参数方式外,也可以使用交互的方式 |
455 | kill -9 <进程号> | 409 | sudo certbot certonly</code></pre> |
456 | |||
457 | # 重启nginx | ||
458 | nginx</code></pre> | ||
459 | <p>而后就可以申请证书辣!</p> | ||
460 | <pre><code class="language-bash"># 申请证书 | ||
461 | sudo certbot certonly --standalone --email example@qq.com -d code.player.com</code></pre> | ||
462 | <p>申请完成后会展示证书的存储路径,记下来,后边会用到。比如:</p> | 410 | <p>申请完成后会展示证书的存储路径,记下来,后边会用到。比如:</p> |
463 | <p>SSL 证书 /etc/letsencrypt/live/code.play.com/fullchain.pem;</p> | 411 | <p>SSL 证书 /etc/letsencrypt/live/code.player.com/fullchain.pem;</p> |
464 | <p>SSL 证书秘钥 /etc/letsencrypt/live/code.play.com/privkey.pem;</p> | 412 | <p>SSL 证书秘钥 /etc/letsencrypt/live/code.player.com/privkey.pem;</p> |
465 | <h3 id="修改添加配置">修改、添加配置</h3> | 413 | <h3 id="修改添加配置">修改、添加配置</h3> |
466 | <p>Nginx | 414 | <p>Nginx |
467 | 默认下载目录在<code>/etc/nginx</code>,该目录下有个<code>.conf</code>文件,但不建议直接改,建议在<code>/etc/nginx/conf.d</code>文件夹下为每个需要转发的域名添加单独的设置文件,这样方便管理和修改。运行过程中 | 415 | 默认下载目录在<code>/etc/nginx</code>,该目录下有个<code>.conf</code>文件,但不建议直接改,建议在<code>/etc/nginx/conf.d</code>文件夹下为每个需要转发的域名添加单独的设置文件,这样方便管理和修改。运行过程中 |
468 | Nginx 会自动将该目录下所有配置文件全部加载的,不需要手动设置。</p> | 416 | Nginx 会自动将该目录下所有配置文件全部加载的,不需要手动设置。</p> |
469 | <p>在<code>/etc/nginx/conf.d</code>中,新建文件<code>vscode.conf</code>,内容如下:</p> | 417 | <p>在<code>/etc/nginx/conf.d</code>中,新建文件<code>vscode.conf</code>,内容如下:</p> |
470 | <pre><code class="language-plaintext">server | 418 | <pre><code>server |
471 | { | 419 | { |
420 | # 这段是一个常用的http重定向到https的方法,用了都说好 | ||
472 | listen 80; | 421 | listen 80; |
473 | server_name code.player.com; | 422 | server_name code.player.com; |
474 | return 301 https://$host$request_uri; | 423 | return 301 https://$host$request_uri; |
@@ -478,21 +427,102 @@ server | |||
478 | server_name code.player.com; | 427 | server_name code.player.com; |
479 | listen 443 ssl; | 428 | listen 443 ssl; |
480 | 429 | ||
430 | # 证书路径 | ||
481 | ssl_certificate /etc/letsencrypt/live/code.player.com/fullchain.pem; | 431 | ssl_certificate /etc/letsencrypt/live/code.player.com/fullchain.pem; |
482 | ssl_certificate_key /etc/letsencrypt/live/code.player.com/privkey.pem; | 432 | ssl_certificate_key /etc/letsencrypt/live/code.player.com/privkey.pem; |
483 | 433 | ||
434 | # 这是对于code-server做出的设置 | ||
484 | location / { | 435 | location / { |
485 | proxy_pass http://127.0.0.1:8080;# 这里的端口号要和code-server指定的端口号一致,我是8080 | 436 | proxy_pass http://127.0.0.1:8080; # 这里的端口号要和code-server指定的端口号一致,我是8080 |
486 | proxy_set_header Host $host; | 437 | proxy_set_header Host $host; |
487 | proxy_set_header Upgrade $http_upgrade; | 438 | proxy_set_header Upgrade $http_upgrade; |
488 | proxy_set_header Connection upgrade; | 439 | proxy_set_header Connection upgrade; |
489 | proxy_set_header Accept-Encoding gzip; | 440 | proxy_set_header Accept-Encoding gzip; |
490 | } | 441 | } |
442 | |||
443 | # 如果文件对应的是webroot服务(也就是一个文件夹下存放着指定网址的网页) | ||
444 | # 需要将上边的配置修改为 | ||
445 | # location / { | ||
446 | # root /var/myweb/; # 这里是文件夹路径 | ||
447 | # index index.html; # 这里是每个文件夹下默认打开的文件 | ||
448 | # } | ||
491 | }</code></pre> | 449 | }</code></pre> |
492 | <p>亲测有效。</p> | ||
493 | <p>编写完文件之后,需要重新加载配置文件(不必重启 Nginx):</p> | 450 | <p>编写完文件之后,需要重新加载配置文件(不必重启 Nginx):</p> |
494 | <pre><code class="language-bash">nginx -s reload</code></pre> | 451 | <pre><code>sudo nginx -s reload</code></pre> |
495 | <p>但在使用过程中,一定记得把域名换成自己的域名、证书路径换成自己的证书路径、端口换成自己在 code-server 中指定的端口!</p> | 452 | <p>再次提醒:一定记得把域名换成自己的域名、证书路径换成自己的证书路径、端口换成自己在提供服务的应用(如 code-server)中指定的端口!</p> |
453 | <h1 id="服务器上的文件浏览">服务器上的文件浏览</h1> | ||
454 | <p>我们已经学会了设置 nginx 的 webroot,但也有很多情况我们需要能在各种设备上直接地看到服务器上有哪些文件,既能方便文件传输,也能覆盖对大多数只需要看不需要改的场景。</p> | ||
455 | <p>python3 为我们提供了一个简单的 http 文件浏览服务,默认安装目录为<code>/usr/lib/python3.x/http/server.py</code>,一般情况下我们可以直接运行它:</p> | ||
456 | <pre><code>python3 -m http.server</code></pre> | ||
457 | <p>但是 python 提供的这份文件存在一些问题:</p> | ||
458 | <ul> | ||
459 | <li>响应头没有规定文件编码,让浏览器自己看着办,容易出现乱码</li> | ||
460 | <li>里边选择性忽略了一些文件、文件夹</li> | ||
461 | </ul> | ||
462 | <p>于是我在给定的基础上做了一些小修改,修改版的文件在<a href="https://www.qin-juan-ge-zhu.top/code/linux/httpmyserver.html">这里</a>。 | ||
463 | </p> | ||
464 | <h1 id="git-远程存储">git 远程存储</h1> | ||
465 | <p>君不闻《西游记》有云:</p> | ||
466 | <blockquote> | ||
467 | <p>争名夺利几时休,早起迟眠不自由。</p> | ||
468 | <p>骑着驴骡思骏马,官居宰相望王侯。</p> | ||
469 | </blockquote> | ||
470 | <p>完成了代码环境的配置,我就在想,要是有一个自己的远程 git 托管环境该多好?那就干!</p> | ||
471 | <p>我的详细配置方法写在<a href="https://www.qin-juan-ge-zhu.top/code/linux/gitserver.html">这里</a>里了。</p> | ||
472 | <h1 id="游戏">游戏</h1> | ||
473 | <blockquote> | ||
474 | <p>All work no play makes Jack a doll boy.</p> | ||
475 | </blockquote> | ||
476 | <p>经过了上边的配置,写代码的控制基本完善了,但是服务器怎么能只拿来干活呢?没有游戏怎么行?我们要快乐!</p> | ||
477 | <p>目前我装了三个游戏:</p> | ||
478 | <ul> | ||
479 | <li>俄罗斯方块,装的是<code>tint</code></li> | ||
480 | <li>扫雷,装的是<code>minebash</code></li> | ||
481 | <li>贪吃蛇,找不到力(允悲</li> | ||
482 | </ul> | ||
483 | <p>安装方法网上都有,就不多说了。玩得愉快!</p> | ||
484 | <h1 id="开机运行脚本">开机运行脚本</h1> | ||
485 | <p>经过了以上的设置,我的服务器已经有了许多服务,特别是每个计算机人电脑里必然需要养的一只宠物——小蓝猫 | ||
486 | clash。但是每次重启系统之后,这许多服务都需要自己手动拉起的话未免太过麻烦了,所以我们需要一个在系统加载完成后就能自动以 root 身份执行的脚本。经过查询,我找到了处理办法。</p> | ||
487 | <p>在 Linux 系统下,开机启动一般使用的是<code>/etc/rc.local</code>文件(但也有很多发行版不再使用这种操作方式)。ubuntu20.04 系统已经默认安装了 | ||
488 | rc-local.service 服务,但是不知什么原因系统把这个服务给“隐蔽”了,所以如果不做一番操作是无法使用的。</p> | ||
489 | <pre><code># 以下所有命令需要root身份执行 | ||
490 | |||
491 | cp /usr/lib/systemd/system/rc-local.service /etc/systemd/system/ | ||
492 | # 修改rc-local.service文件,在文件末尾添加以下内容(注意删去前边的注释符): | ||
493 | # [Install] | ||
494 | # WantedBy=multi-user.target | ||
495 | |||
496 | # 创建rc.local文件,带上shebang行 | ||
497 | echo '#!/bin/bash' > /etc/rc.local | ||
498 | # 修改rc.local文件权限 | ||
499 | chmod +x /etc/rc.local | ||
500 | |||
501 | # 上述步骤完成后,注意不要急于写rc.local脚本,我们需要先启动rc-local.service服务 | ||
502 | systemctl start rc-local.service | ||
503 | systemctl enable rc-local.service | ||
504 | # 上述两条命令运行正常时,系统需要重启 | ||
505 | reboot | ||
506 | |||
507 | # 重启完成后,检查rc-local.service服务是否正常运行 | ||
508 | systemctl status rc-local.service | ||
509 | |||
510 | # 一切正常,就可以编写rc.local脚本的内容了 | ||
511 | # 需要注意的是,该脚本以root身份运行,所以不需要sudo | ||
512 | # 如果部分命令需要以其他用户身份运行,可以使用su命令,如: | ||
513 | # su -l <username> -c <要运行的命令> | ||
514 | # 如果要运行的是脚本(不建议,没必要),注意对应参数 | ||
515 | # 文件的最后,可以用exec &> /var/log/rc-local.log将脚本的输出重定向到日志文件中</code></pre> | ||
516 | <p>除此之外,我们可能还需要为所有用户默认开启魔法。这个功能在开机脚本和<code>/etc/profile</code>中都无法实现,应当放在<code>/etc/environment</code>中。在其中添加以下内容: | ||
517 | </p> | ||
518 | <pre><code>PATH="/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin:/usr/games:/usr/local/games:/snap/bin" | ||
519 | http_proxy=http://127.0.0.1:7890 | ||
520 | https_proxy=http://127.0.0.1:7890 | ||
521 | no_proxy=127.0.0.1,localhost | ||
522 | HTTP_PROXY=http://127.0.0.1:7890 | ||
523 | HTTPS_PROXY=http://127.0.0.1:7890 | ||
524 | NO_PROXY=127.0.0.1,localhost</code></pre> | ||
525 | <p>该文件中的所有设置对于所有用户均有效,着实省去了每次开机、切换用户都要手动挂代理的麻烦。</p> | ||
496 | <h1 id="碎碎念">碎碎念</h1> | 526 | <h1 id="碎碎念">碎碎念</h1> |
497 | <blockquote> | 527 | <blockquote> |
498 | <p>王子猷(徽之)居山阴,夜大雪,眠觉,开室命酌酒,四望皎然。因起彷徨。咏左思《招隐诗》,忽忆戴安道(逵)。时戴在剡,即便夜乘小船就之。经宿方至,造门不前而返。人问其故,王曰:“吾本乘兴而行,兴尽而返,何必见戴!” | 528 | <p>王子猷(徽之)居山阴,夜大雪,眠觉,开室命酌酒,四望皎然。因起彷徨。咏左思《招隐诗》,忽忆戴安道(逵)。时戴在剡,即便夜乘小船就之。经宿方至,造门不前而返。人问其故,王曰:“吾本乘兴而行,兴尽而返,何必见戴!” |
@@ -501,8 +531,8 @@ server | |||
501 | </blockquote> | 531 | </blockquote> |
502 | <p>一时兴起跟着黄四郎同志搞了前端,又一时兴起薅资本主义羊毛搞服务器,甚至现在写这篇文档也是一时兴起。可能我就是这样一个人吧,想到哪里做到哪里,全是为了快乐。</p> | 532 | <p>一时兴起跟着黄四郎同志搞了前端,又一时兴起薅资本主义羊毛搞服务器,甚至现在写这篇文档也是一时兴起。可能我就是这样一个人吧,想到哪里做到哪里,全是为了快乐。</p> |
503 | <p>这篇文档是对一个月以来搞云服务器的配置的一个小小总结,本来以为没多少东西,结果写了整整一下午到现在,一个 Markdown 文档,手写居然达到了 900 多行、将近 40KB,令我也大吃一惊。</p> | 533 | <p>这篇文档是对一个月以来搞云服务器的配置的一个小小总结,本来以为没多少东西,结果写了整整一下午到现在,一个 Markdown 文档,手写居然达到了 900 多行、将近 40KB,令我也大吃一惊。</p> |
504 | <p>也许我之后还会继续鼓捣我的服务器,那时候就继续写下去吧。现在也许该考虑考虑期末考试了(笑。</p> | 534 | <p>也许我之后还会继续鼓捣我的服务器,那时候就继续写下去吧。现在该考虑考虑期末考试了(笑。</p> |
505 | <p class="time">2023.5.30傍晚</p> | 535 | <p class="time">2023.5.30傍晚<br>2024.1.3</p> |
506 | <script src="https://www.qin-juan-ge-zhu.top/common/js/comment.js"></script> | 536 | <script src="https://www.qin-juan-ge-zhu.top/common/js/comment.js"></script> |
507 | </div> | 537 | </div> |
508 | </div> | 538 | </div> |
diff --git a/code/linux/aliyun_cloud.md b/code/linux/aliyun_cloud.md index 5bd4857..d059571 100644 --- a/code/linux/aliyun_cloud.md +++ b/code/linux/aliyun_cloud.md | |||
@@ -18,17 +18,17 @@ | |||
18 | 18 | ||
19 | 另外,由于是一个临时账号,我个人建议另找一个浏览器做实验,方便和自己现在用的帐户/ECS 实例区别开来。 | 19 | 另外,由于是一个临时账号,我个人建议另找一个浏览器做实验,方便和自己现在用的帐户/ECS 实例区别开来。 |
20 | 20 | ||
21 | # 服器配置 | 21 | ## 件配置 |
22 | 22 | ||
23 | 服务器硬件配置应该都是固定的吧?CPU 双核,内存 2G,硬盘 40G,带宽 1MBps。我个人觉得这个配置还是挺不错的,毕竟是白嫖,要求也不能太高了不是? | 23 | 服务器硬件配置应该都是固定的吧?CPU 双核,内存 2G,硬盘 40G,带宽 1MBps。我个人觉得这个配置还是挺不错的,毕竟是白嫖,要求也不能太高了不是? |
24 | 24 | ||
25 | 操作系统方面,由于个人比较喜欢纯命令行操作,所以选择了 Ubuntu22.04 LTS,我的虚拟机也是这个版本,相对而言会比其他发行版更熟悉一些。 | 25 | 操作系统方面,由于个人比较喜欢纯命令行操作,所以选择了 Ubuntu22.04 LTS,我的虚拟机也是这个版本,相对而言会比其他发行版更熟悉一些。 |
26 | 26 | ||
27 | ## 服务器初始化 | 27 | # 服务器初始化 |
28 | 28 | ||
29 | 在跟着创建实例的要求一步步做好之后,我们就拥有了一台属于自己的服务器了,并已经有了根用户 root。接下来,我们需要做一些初始化的工作。 | 29 | 在跟着创建实例的要求一步步做好之后,我们就拥有了一台属于自己的服务器了,并已经有了根用户 root。接下来,我们需要做一些初始化的工作。 |
30 | 30 | ||
31 | ### 创建新用户 | 31 | ## 创建新用户 |
32 | 32 | ||
33 | 众所周知,**为了安全起见,我们需要一个非 root 但具有 sudo 权限的用户来进行日常操作**。这里我创建了一个名为<kbd>player</kbd>的用户并设置了密码,然后将其加入 sudo 组。 | 33 | 众所周知,**为了安全起见,我们需要一个非 root 但具有 sudo 权限的用户来进行日常操作**。这里我创建了一个名为<kbd>player</kbd>的用户并设置了密码,然后将其加入 sudo 组。 |
34 | 34 | ||
@@ -48,37 +48,10 @@ sudo apt install vim | |||
48 | 48 | ||
49 | cd /etc | 49 | cd /etc |
50 | sudo vim sudoers | 50 | sudo vim sudoers |
51 | # 找到root ALL=(ALL:ALL) ALL,按o新开一行,然后写一份一样的,把root改成player | 51 | # 找到root ALL=(ALL:ALL) ALL, |
52 | # ESC,输入:wq保存退出 | 52 | # 开一然后写一份一样的,把root改成player,保存退出 |
53 | ``` | 53 | ``` |
54 | 54 | ||
55 | 这时我们会发现一个很操蛋的事情,“为啥命令提示符之前不显示当前用户名和当前路径嘞?这岂不是每次看路径都需要<kbd>pwd</kbd>吗?烦不烦啊!”别急,问题很好修改。 | ||
56 | |||
57 | ```bash | ||
58 | cd /home/player | ||
59 | vim .bashrc | ||
60 | # 找到PS1=,将其修改为PS1="\u@\h:\w\$" | ||
61 | # 按ESC,输入:wq保存退出 | ||
62 | ``` | ||
63 | |||
64 | 修改退出之后似乎一切都没有变化。我们需要重启一下服务器。 | ||
65 | |||
66 | ```bash | ||
67 | sudo reboot | ||
68 | ``` | ||
69 | |||
70 | 重新连接之后,也许还是没有变化,我就是这样的。后来在处理其它问题的时候,我发现了根本原因所在:我 TM 创建用户的时候忘了加<kbd>-s</kbd>参数,导致新用户的默认 shell 是<kbd>sh</kbd>而不是<kbd>bash</kbd>。所以,我们需要修改一下新用户的默认 shell。 | ||
71 | |||
72 | ```bash | ||
73 | # 查看当前用户的默认shell | ||
74 | echo $SHELL | ||
75 | |||
76 | # 修改默认shell | ||
77 | sudo chsh -s /bin/bash player | ||
78 | ``` | ||
79 | |||
80 | 这时再重启服务器,就会发现,一切都正常了,一切是那么美好。 | ||
81 | |||
82 | ## 安装软件 | 55 | ## 安装软件 |
83 | 56 | ||
84 | 经历了上边的配置,现在我们拥有了一台可以正常使用的服务器。服务器是为了什么?开发!开发!开发!所以,我们需要安装一些开发所需的软件。 | 57 | 经历了上边的配置,现在我们拥有了一台可以正常使用的服务器。服务器是为了什么?开发!开发!开发!所以,我们需要安装一些开发所需的软件。 |
@@ -111,17 +84,182 @@ sudo apt install make cmake | |||
111 | # 剩下需要安装的软件,可以自行搜索 | 84 | # 剩下需要安装的软件,可以自行搜索 |
112 | ``` | 85 | ``` |
113 | 86 | ||
114 | ## 配置 ssh | 87 | ## 简单美化一下命令行 |
88 | |||
89 | 配到这里,我们会发现一个很操蛋的事情,“为啥命令提示符之前不显示当前用户名和当前路径嘞?这岂不是每次看路径都需要<kbd>pwd</kbd>吗?烦不烦啊!”别急,问题很好修改。 | ||
90 | |||
91 | ```bash | ||
92 | cd ~ | ||
93 | vim .bashrc | ||
94 | # 找到PS1=,将其修改为PS1="\u@\h:\w\$" | ||
95 | |||
96 | # 加载.bashrc | ||
97 | source .bashrc | ||
98 | ``` | ||
99 | |||
100 | 改好之后也许还是没有变化,我就是这样的。后来在处理其它问题的时候,我发现了根本原因所在:我 TM 创建用户的时候忘了加<kbd>-s</kbd>参数,导致新用户的默认 shell 是<kbd>sh</kbd>而不是<kbd>bash</kbd>。所以,我们需要修改一下新用户的默认 shell。 | ||
101 | |||
102 | ```bash | ||
103 | # 查看当前用户的默认shell | ||
104 | echo $SHELL | ||
105 | |||
106 | # 修改默认shell | ||
107 | sudo chsh -s /bin/bash player | ||
108 | ``` | ||
109 | |||
110 | 这时再重新加载,就会发现,一切都正常了,一切是那么美好。 | ||
111 | |||
112 | 除此之外,我们也许想要调整命令行显示的用户名、路径之类内容的颜色,会想让命令行像 git bash 一样当我们进入 git 仓库的时候显示当前在什么分支。针对这两项需求,我修改了一下我的.bashrc 文件,现在其内容如下: | ||
113 | |||
114 | ```bash | ||
115 | # ~/.bashrc: executed by bash(1) for non-login shells. | ||
116 | # see /usr/share/doc/bash/examples/startup-files (in the package bash-doc) | ||
117 | # for examples | ||
118 | |||
119 | # If not running interactively, don't do anything | ||
120 | case $- in | ||
121 | *i*) ;; | ||
122 | *) return;; | ||
123 | esac | ||
124 | |||
125 | # don't put duplicate lines or lines starting with space in the history. | ||
126 | # See bash(1) for more options | ||
127 | HISTCONTROL=ignoreboth | ||
128 | |||
129 | # append to the history file, don't overwrite it | ||
130 | shopt -s histappend | ||
131 | |||
132 | # for setting history length see HISTSIZE and HISTFILESIZE in bash(1) | ||
133 | HISTSIZE=1000 | ||
134 | HISTFILESIZE=2000 | ||
135 | |||
136 | # check the window size after each command and, if necessary, | ||
137 | # update the values of LINES and COLUMNS. | ||
138 | shopt -s checkwinsize | ||
139 | |||
140 | # If set, the pattern "**" used in a pathname expansion context will | ||
141 | # match all files and zero or more directories and subdirectories. | ||
142 | #shopt -s globstar | ||
143 | |||
144 | # make less more friendly for non-text input files, see lesspipe(1) | ||
145 | [ -x /usr/bin/lesspipe ] && eval "$(SHELL=/bin/sh lesspipe)" | ||
146 | |||
147 | # set variable identifying the chroot you work in (used in the prompt below) | ||
148 | if [ -z "${debian_chroot:-}" ] && [ -r /etc/debian_chroot ]; then | ||
149 | debian_chroot=$(cat /etc/debian_chroot) | ||
150 | fi | ||
151 | |||
152 | # set a fancy prompt (non-color, unless we know we "want" color) | ||
153 | case "$TERM" in | ||
154 | xterm-color|*-256color) color_prompt=yes;; | ||
155 | esac | ||
156 | |||
157 | # uncomment for a colored prompt, if the terminal has the capability; turned | ||
158 | # off by default to not distract the user: the focus in a terminal window | ||
159 | # should be on the output of commands, not on the prompt | ||
160 | #force_color_prompt=yes | ||
161 | |||
162 | if [ -n "$force_color_prompt" ]; then | ||
163 | if [ -x /usr/bin/tput ] && tput setaf 1 >&/dev/null; then | ||
164 | # We have color support; assume it's compliant with Ecma-48 | ||
165 | # (ISO/IEC-6429). (Lack of such support is extremely rare, and such | ||
166 | # a case would tend to support setf rather than setaf.) | ||
167 | color_prompt=yes | ||
168 | else | ||
169 | color_prompt= | ||
170 | fi | ||
171 | fi | ||
172 | |||
173 | # 简单修改一下颜色 | ||
174 | if [ "$color_prompt" = yes ]; then | ||
175 | PS1='${debian_chroot:+($debian_chroot)}\[\033[01;32m\]\u@\h\[\033[00m\]:\[\033[01;34m\]\[\033[00m\]' | ||
176 | else | ||
177 | PS1='${debian_chroot:+($debian_chroot)}\u@\h:' | ||
178 | fi | ||
179 | |||
180 | # 如果是git仓库,在命令行提示符之前用不同的颜色显示当前分支 | ||
181 | PS1="$PS1"'\[\033[33m\]' # change to yellow | ||
182 | PS1="$PS1"'\w' # pwd | ||
183 | PS1="$PS1"'\[\033[36m\]' # change color to cyan | ||
184 | PS1="$PS1"'`__git_ps1`' # git branch | ||
185 | PS1="$PS1"'\[\033[0m\]' # change color | ||
186 | #PS1="$PS1"'\n' # new line | ||
187 | PS1="$PS1"'\$ ' # prompt: always $ | ||
188 | unset color_prompt force_color_prompt | ||
189 | |||
190 | # If this is an xterm set the title to user@host:dir | ||
191 | case "$TERM" in | ||
192 | xterm*|rxvt*) | ||
193 | PS1="\[\e]0;${debian_chroot:+($debian_chroot)}\u@\h: \w\a\]$PS1" | ||
194 | ;; | ||
195 | *) | ||
196 | ;; | ||
197 | esac | ||
198 | |||
199 | # enable color support of ls and also add handy aliases | ||
200 | if [ -x /usr/bin/dircolors ]; then | ||
201 | test -r ~/.dircolors && eval "$(dircolors -b ~/.dircolors)" || eval "$(dircolors -b)" | ||
202 | alias ls='ls --color=auto' | ||
203 | #alias dir='dir --color=auto' | ||
204 | #alias vdir='vdir --color=auto' | ||
205 | |||
206 | alias grep='grep --color=auto' | ||
207 | alias fgrep='fgrep --color=auto' | ||
208 | alias egrep='egrep --color=auto' | ||
209 | fi | ||
210 | |||
211 | # colored GCC warnings and errors | ||
212 | #export GCC_COLORS='error=01;31:warning=01;35:note=01;36:caret=01;32:locus=01:quote=01' | ||
213 | |||
214 | # some more ls aliases | ||
215 | alias ll='ls -alF' | ||
216 | alias la='ls -A' | ||
217 | alias l='ls -CF' | ||
218 | |||
219 | # Add an "alert" alias for long running commands. Use like so: | ||
220 | # sleep 10; alert | ||
221 | alias alert='notify-send --urgency=low -i "$([ $? = 0 ] && echo terminal || echo error)" "$(history|tail -n1|sed -e '\''s/^\s*[0-9]\+\s*//;s/[;&|]\s*alert$//'\'')"' | ||
222 | |||
223 | # Alias definitions. | ||
224 | # You may want to put all your additions into a separate file like | ||
225 | # ~/.bash_aliases, instead of adding them here directly. | ||
226 | # See /usr/share/doc/bash-doc/examples in the bash-doc package. | ||
227 | |||
228 | if [ -f ~/.bash_aliases ]; then | ||
229 | . ~/.bash_aliases | ||
230 | fi | ||
231 | |||
232 | # enable programmable completion features (you don't need to enable | ||
233 | # this, if it's already enabled in /etc/bash.bashrc and /etc/profile | ||
234 | # sources /etc/bash.bashrc). | ||
235 | if ! shopt -oq posix; then | ||
236 | if [ -f /usr/share/bash-completion/bash_completion ]; then | ||
237 | . /usr/share/bash-completion/bash_completion | ||
238 | elif [ -f /etc/bash_completion ]; then | ||
239 | . /etc/bash_completion | ||
240 | fi | ||
241 | fi | ||
242 | echo This\ is\ .bashrc. | ||
243 | |||
244 | alias pip='python -m pip' | ||
245 | # 设置bash使用vim模式,允许使用0、dw之类常用vim快捷键,方便命令编辑 | ||
246 | set -o vi | ||
247 | # 有时候我们需要一些魔法 | ||
248 | export http_proxy=http://127.0.0.1:7890 | ||
249 | export https_proxy=http://127.0.0.1:7890 | ||
250 | ``` | ||
251 | |||
252 | # 配置 ssh | ||
115 | 253 | ||
116 | 用着用着,就会发现:我为啥一直在浏览器里搞命令行?要是能直接在本地的命令行里搞就好了。当然,阿里云能满足我们。 | 254 | 用着用着,就会发现:我为啥一直在浏览器里搞命令行?要是能直接在本地的命令行里搞就好了。当然,阿里云能满足我们。 |
117 | 255 | ||
118 | ### ssh 简介 | 256 | ## ssh 简介 |
119 | 257 | ||
120 | 什么是 ssh?ssh 是 Secure Shell 的缩写,中文名为安全外壳协议。ssh 是一种加密的网络协议,可以在不安全的网络中为网络服务提供安全的传输环境。ssh 是目前较可靠,专为远程登录会话和其他网络服务提供安全性的协议。利用 ssh 协议可以有效防止远程管理过程中的信息泄露问题。通过 ssh 可以对所有传输的数据进行加密,也能够防止 DNS 欺骗和 IP 欺骗。ssh 还支持在两个远程主机之间进行数据的传输。 | 258 | 什么是 ssh?ssh 是 Secure Shell 的缩写,中文名为安全外壳协议。ssh 是一种加密的网络协议,可以在不安全的网络中为网络服务提供安全的传输环境。ssh 是目前较可靠,专为远程登录会话和其他网络服务提供安全性的协议。利用 ssh 协议可以有效防止远程管理过程中的信息泄露问题。通过 ssh 可以对所有传输的数据进行加密,也能够防止 DNS 欺骗和 IP 欺骗。ssh 还支持在两个远程主机之间进行数据的传输。 |
121 | 259 | ||
122 | 说了也记不住,直接开干! | 260 | 说了也记不住,直接开干! |
123 | 261 | ||
124 | ### 配置安全组规则 | 262 | ## 配置安全组规则 |
125 | 263 | ||
126 | 在阿里云控制台左上角点击,选择`云服务器ECS`,会进入云服务器实例控制界面。在左侧菜单栏寻找`安全组` ,添加安全组规则。按要求添加新的安全组规则,开放 TCP 协议对应的 22 端口。(其实可以都开了,但是毕竟不安全,建议后续按需开放。) | 264 | 在阿里云控制台左上角点击,选择`云服务器ECS`,会进入云服务器实例控制界面。在左侧菜单栏寻找`安全组` ,添加安全组规则。按要求添加新的安全组规则,开放 TCP 协议对应的 22 端口。(其实可以都开了,但是毕竟不安全,建议后续按需开放。) |
127 | 265 | ||
@@ -132,7 +270,7 @@ sudo apt install make cmake | |||
132 | - 将上述.pem 文件移动到家目录`C:\用户\当前用户名`的<kbd>.ssh</kbd>文件夹下 | 270 | - 将上述.pem 文件移动到家目录`C:\用户\当前用户名`的<kbd>.ssh</kbd>文件夹下 |
133 | - 在<kbd>.ssh</kbd>文件夹下创建文件<kbd>config</kbd>,内容如下: | 271 | - 在<kbd>.ssh</kbd>文件夹下创建文件<kbd>config</kbd>,内容如下: |
134 | 272 | ||
135 | ```txt | 273 | ```plaintext |
136 | # 给服务器起的别名,这里是aliyun | 274 | # 给服务器起的别名,这里是aliyun |
137 | Host aliyun | 275 | Host aliyun |
138 | # IP地址,这里以127.0.0.1代替,但真实操作需要换成服务器的公网ip,在控制台就能看到 | 276 | # IP地址,这里以127.0.0.1代替,但真实操作需要换成服务器的公网ip,在控制台就能看到 |
@@ -156,255 +294,21 @@ ssh aliyun | |||
156 | 294 | ||
157 | 既然我们选择了在命令行里使用,编辑文件当然是比较常用 vim 了。当然 neovim 与其他命令行编辑器也很好,但我不熟悉,请参照网上其他教程进行配置。 | 295 | 既然我们选择了在命令行里使用,编辑文件当然是比较常用 vim 了。当然 neovim 与其他命令行编辑器也很好,但我不熟悉,请参照网上其他教程进行配置。 |
158 | 296 | ||
159 | 我的 vim 学习与配置[链接](https://www.qin-juan-he-zhu.top/code/linux/vim.html)中有如下一些基本配置: | 297 | 我的 vim 学习与配置[链接](https://www.qin-juan-ge-zhu.top/code/linux/vim.html)中有如下一些基本配置: |
160 | 298 | ||
161 | - 同时开启绝对行号与相对行号,则当前行显示真实行号,其他行显示相对于当前行为上/下第几行,方便跳转。 | ||
162 | - 代码中常常需要的`()[]{}`自动匹配 | ||
163 | - 自动缩进和智能缩进,但似乎没啥鸟用 | ||
164 | - vim 自带的代码高亮,感觉还是相当不错的 | 299 | - vim 自带的代码高亮,感觉还是相当不错的 |
300 | - 代码中常常需要的`()[]{}`自动匹配 | ||
301 | - 同时开启绝对行号与相对行号,则当前行显示真实行号,其他行显示相对于当前行为上/下第几行,方便跳转。 | ||
302 | - 代码按语法缩进,并且每次保存或退出时自动格式化 | ||
165 | - 当前行/当前列各给一个颜色,因为有的平台光标显示很不明显,正常插入模式一般也不好确定光标位置,这样方便知道光标在哪。 | 303 | - 当前行/当前列各给一个颜色,因为有的平台光标显示很不明显,正常插入模式一般也不好确定光标位置,这样方便知道光标在哪。 |
166 | - 自动折行,即当前行太长了会被折回来显示,但还是同一行,方便一次性看完整。 | 304 | - 自动折行,即当前行太长了会被折回来显示,但还是同一行,方便一次性看完整。 |
167 | - 一些快捷键,如<kbd>ta</kbd>到行末(替代够不到的<kbd>$</kbd>),<kbd>end</kbd>来到文件末(替代难记的<kbd>G</kbd>),还有<kbd>up/ne</kbd>上/下移动 20 行等。 | 305 | - 一些快捷键,如<kbd>L</kbd>到行末(替代够不到的<kbd>$</kbd>),还有<kbd>K/J</kbd>上/下移动 10 行等。 |
168 | - 自动识别文件类型,并对`*.html, *.c, *.cpp, *.python`等文件在保存<kbd>:w<Enter></kbd>时自动格式化,在正常或选择模式下<kbd>Ctrl+p</kbd>一键注释(别骂了别骂了,<kbd>Ctrl+/</kbd>我属实是设置不出来) | 306 | - 基本的状态栏设置,包括文件名、git 分支、文件编码、文件类型、总行数等信息,方便查看。 |
169 | 307 | - 自动识别文件类型,并对`*.html, *.c, *.cpp, *.python`等文件在保存<kbd>:w<Enter></kbd>时自动格式化 | |
170 | ## 插件 | 308 | - 插件和其他重重 |
171 | 309 | ||
172 | 这里使用了几个插件,插件管理目前用的是评价一般的`Vundle`。个人评价,下载插件啥的属实是慢,每次都报错报到怀疑人生,害得我都得手动安装然后在里边添加,但是别的目前还不会使。气死偶勒! | 310 | 这里使用了几个插件,插件管理目前用的是评价一般的`Vundle`。个人评价,下载插件啥的属实是慢,每次都报错报到怀疑人生,害得我都得手动安装然后在里边添加,但是别的目前还不会使。气死偶勒! |
173 | 311 | ||
174 | 这里需要注意的是,上文提到我喜欢把有用的东西放在`~/useful`下,但是如果直接这样写路径,其他用户如 root 等使用时候就会报错,人家的主目录第下没这个东西!所以,在写路径的位置,切记要写绝对路径!!! | ||
175 | |||
176 | # 游戏 | ||
177 | |||
178 | > All work no play makes Jack a doll boy. | ||
179 | |||
180 | 经过了上边的配置,写代码的控制基本完善了,但是服务器怎么能只拿来干活呢?没有游戏怎么行?我们要快乐! | ||
181 | |||
182 | 目前我的服务器上装了三个游戏 | ||
183 | |||
184 | - 俄罗斯方块,装的是`tint` | ||
185 | - 扫雷,装的是`miebash` | ||
186 | - 贪吃蛇,找不到力(允悲 | ||
187 | |||
188 | 安装方法网上都有,就不多说了。玩得愉快! | ||
189 | |||
190 | # git 远程存储 | ||
191 | |||
192 | 君不闻,《西游记》有云: | ||
193 | |||
194 | > 争名夺利几时休,早起迟眠不自由。 | ||
195 | > | ||
196 | > 骑着驴骡思骏马,官居宰相望王侯。 | ||
197 | |||
198 | 完成了代码环境的配置,我就在想,要是有一个自己的远程 git 存储库该多好? | ||
199 | |||
200 | 说干就干,我找到了 gitlab 安装教程,信心满满,谁知"先帝创业未半而中道崩殂"。 | ||
201 | |||
202 | ## gitlab? | ||
203 | |||
204 | 我按照网上的教程,更新了清华镜像源,下载安装 gitlab。孰料一次一次在安装过程卡死,而且情况十分严重,CPU | ||
205 | 占用率长期维持在 50%以上,内存更是飙升到 90%,现在不仅安装进行不下去了,甚至一切操作都不认识,包括<kbd>Ctrl+C</kbd>想要杀死当前进程的请求也会石沉大海。没办法,只得强制重启。 | ||
206 | |||
207 | 如是反复若干次,最终死心,将 gitlab 彻底从云服务器卸载,"革职为民,永不叙用"。 | ||
208 | |||
209 | 经过上网查找,原因果然出现在配置上。**gitlab 推荐的最小内存是 4G,但是目前我们只有 2G,内存爆满也就理所当然了。**没办法,期限之内不能更改配置,就算能更改,我都穷到薅资本主义羊毛了,还能有钱升级配置?笑死。 | ||
210 | |||
211 | ## 就这? | ||
212 | |||
213 | 就因为这个问题,我们就不干了?这是不行滴,小同志。 | ||
214 | |||
215 | 不久之后,黄四郎同志发来了一篇博客:[最简单的 git 服务器](https://www.ruanyifeng.com/blog/2022/10/git-server.html) | ||
216 | |||
217 | 看起来似乎不错,但是失之简略。幸好,我在 git 官方教程[Pro Git](https://git-scm.com/book/zh/v2)上找到了另一部分,两个拼一拼、试一试,最终成功了。 | ||
218 | |||
219 | 注意,本地和远程都需要安装 git,相信能看这里的读者应该已经是安装过了,这里不再赘述。 | ||
220 | |||
221 | ### 前置 | ||
222 | |||
223 | 在创建存储库之前,我想我们应该做些什么。 | ||
224 | |||
225 | 对了,创建新用户吧!按照上述写的步骤创建一个新用户并为之设置密码,可以不用加入 sudo 组,因为我们不需要这个用户来操作服务器,只需要用来存储代码就行了。 | ||
226 | |||
227 | ### 本地存储库 | ||
228 | |||
229 | 本地我们需要一个存储库,如果已有的则可以忽略本部分。 | ||
230 | |||
231 | ```bash | ||
232 | # 创建本地存储库,这里叫test吧 | ||
233 | mkdir test | ||
234 | cd test | ||
235 | git init | ||
236 | ``` | ||
237 | |||
238 | 但是本方法要求本地存储库必须已经有 commit(s)。好办,就写一个常用的`.gitignore`好了。 | ||
239 | |||
240 | ```vim | ||
241 | *.sh | ||
242 | *.bat | ||
243 | *.exe | ||
244 | *.dll | ||
245 | *.so | ||
246 | *.[oa] | ||
247 | *.idea | ||
248 | ``` | ||
249 | |||
250 | 而后,为了方便提交操作,可以再创建一个`push.sh`脚本: | ||
251 | |||
252 | ```bash | ||
253 | #!/bin/bash | ||
254 | git add . | ||
255 | git commit | ||
256 | git push | ||
257 | ``` | ||
258 | |||
259 | 注意,创建`push.sh`一定要在`.gitignore`之后,否则不会被忽略的。 | ||
260 | |||
261 | ### 远程仓库 | ||
262 | |||
263 | 就一句话: | ||
264 | |||
265 | ```bash | ||
266 | # 创建远程仓库,这里也叫test吧 | ||
267 | ssh git@127.0.0.1 git init --bare test.git | ||
268 | ``` | ||
269 | |||
270 | 注意,本处指明的 git 为远程用户名,127.0.0.1 代表云服务器的公网 ip,test.git 为远程仓库名。使用的时候都需要换成自己的。 | ||
271 | |||
272 | ### 本地与远程连接 | ||
273 | |||
274 | 本地和远程都有了,下一步就是建立联系了。 | ||
275 | |||
276 | ```bash | ||
277 | # 本地添加远程 | ||
278 | git remote add origin git@127.0.0.1:test.git | ||
279 | |||
280 | # 此时尚不能直接推送,因为并未指定上游对应分支,需要指定 | ||
281 | git push --set-upstream origin master | ||
282 | |||
283 | # 之后就可以直接推送了 | ||
284 | sh push.sh | ||
285 | ``` | ||
286 | |||
287 | ### 自动化 | ||
288 | |||
289 | 怎么样,操作看起来又臭又长吧?我也这么觉得。所以我写了两份代码,供同时自动创建本地仓库和远程对应仓库并建立所有对应关系之用。 | ||
290 | |||
291 | ```c | ||
292 | /* | ||
293 | * gitadd.c | ||
294 | */ | ||
295 | #include <stdio.h> | ||
296 | #include <stdlib.h> | ||
297 | #include <string.h> | ||
298 | #include <direct.h> | ||
299 | |||
300 | int main() | ||
301 | { | ||
302 | char local_name[100],remote_name[100],tmp[100]; | ||
303 | printf("您好!\n请输入本地仓库名:"); | ||
304 | gets(local_name); | ||
305 | printf("请输入远程仓库名:"); | ||
306 | gets(remote_name); | ||
307 | |||
308 | //远程仓库创建 | ||
309 | strcpy(tmp,"ssh aliyun-git git init --bare "); | ||
310 | strcat(remote_name,".git"); | ||
311 | strcat(tmp,remote_name); | ||
312 | system(tmp); | ||
313 | |||
314 | //本地仓库创建 | ||
315 | mkdir(local_name); | ||
316 | chdir(local_name); | ||
317 | system("git init"); | ||
318 | |||
319 | //本地仓库初始化 | ||
320 | FILE* file; | ||
321 | //编写.gitignore | ||
322 | file=fopen(".gitignore","w"); | ||
323 | fprintf(file,"*.sh\n"); | ||
324 | fprintf(file,"*.bat\n"); | ||
325 | fprintf(file,"*.exe\n"); | ||
326 | fprintf(file,"*.[oa]\n"); | ||
327 | fprintf(file,"*.pyc\n"); | ||
328 | fprintf(file,"__pycache__\n"); | ||
329 | fprintf(file,"*.vscode\n"); | ||
330 | fprintf(file,"*.swp\n"); | ||
331 | fclose(file); | ||
332 | //编写push.sh | ||
333 | file=fopen("push.sh","w"); | ||
334 | fprintf(file,"git add .\n"); | ||
335 | fprintf(file,"git commit\n"); | ||
336 | fprintf(file,"git push"); | ||
337 | fclose(file); | ||
338 | |||
339 | //提交初始化commit | ||
340 | system("git add ."); | ||
341 | system("git commit -m \"Initial commit\""); | ||
342 | strcpy(tmp,"git remote add origin aliyun-git:"); | ||
343 | strcat(tmp,remote_name); | ||
344 | system(tmp); | ||
345 | system("git push --set-upstream origin master"); | ||
346 | system("git push"); | ||
347 | |||
348 | printf("完成!"); | ||
349 | return 0; | ||
350 | } | ||
351 | ``` | ||
352 | |||
353 | ```python | ||
354 | # gitadd.py | ||
355 | import os | ||
356 | |||
357 | if __name__ == '__main__': | ||
358 | print("你好!") | ||
359 | print("请输入本地仓库名:") | ||
360 | local_name = input() | ||
361 | print("请输入远程仓库名:") | ||
362 | remote_name = input() | ||
363 | os.system("ssh aliyun-git git init --bare "+remote_name+".git") | ||
364 | |||
365 | # 创建本地仓库 | ||
366 | os.system("mkdir "+local_name) | ||
367 | os.chdir(local_name) | ||
368 | print("本地仓库已建立!") | ||
369 | print("当前路径为"+os.getcwd()) | ||
370 | |||
371 | # 初始化git仓库 | ||
372 | os.system("git init") | ||
373 | |||
374 | # 以utf8编码编写.gitignore文件 | ||
375 | with open(".gitignore", "w", encoding="utf8") as f: | ||
376 | f.write("*.pyc\n") | ||
377 | f.write("__pycache__/\n") | ||
378 | f.write("*.exe\n") | ||
379 | f.write("*.[oa]\n") | ||
380 | f.write("*.swp\n") | ||
381 | f.write("*.sh\n") | ||
382 | f.write("*.bat\n") | ||
383 | |||
384 | # 编写push.sh文件 | ||
385 | with open("push.sh", "w", encoding="utf8") as f: | ||
386 | f.write("git add .\n") | ||
387 | f.write("git commit\n") | ||
388 | f.write("git push\n") | ||
389 | |||
390 | print("本地仓库初始化完成!") | ||
391 | |||
392 | # 连接远程仓库 | ||
393 | os.system("git add .") | ||
394 | os.system("git commit -m \"Initial commit\"") | ||
395 | os.system("git remote add origin aliyun-git:"+remote_name+".git") | ||
396 | os.system("git push --set-upstream origin master") | ||
397 | os.system("git push") | ||
398 | |||
399 | print("完成!") | ||
400 | ``` | ||
401 | |||
402 | **Attention please:** 本地仓库名和远程仓库名不要加后缀,程序会自动添加。另外,如果要修改,在转换本地当前工作目录的时候,**一定要使用语言自带的调整当前工作目录的函数,不要调用系统命令**,否则会出现编译器/解释器自动将程序作为多线程执行,导致一个线程进去了然后线程当场去世,后续的操作在另一个线程,工作目录根本没发生变化,也就是把原来所在的位置给 git 初始化了,出现奇奇怪怪的错误。 | ||
403 | |||
404 | ### 其他 | ||
405 | |||
406 | 需要明白的是,上述方法在远端创立的**只是一个裸仓库**(即只是我们本地仓库的`.git`文件夹),不包含工作目录,所以不能直接在远端进行操作,需要在本地进行操作,然后推送到远端。但是`git clone`的时候,本地得到的是一个完整的仓库,而其他使用也与普通仓库无异。 | ||
407 | |||
408 | # 来看看在线 VSCode 吧! | 312 | # 来看看在线 VSCode 吧! |
409 | 313 | ||
410 | 本来,一切就这样完美收官、从胜利走向更大的胜利了。坏就坏在我按照五个月前的计划添置了一个 iPad。在最初的设想里它就不单单是当平板做笔记看书用的,而是要充当电脑的编程功能,来在必要的时刻随拿随走,替代我这十几斤重、续航还短的游戏本的。 | 314 | 本来,一切就这样完美收官、从胜利走向更大的胜利了。坏就坏在我按照五个月前的计划添置了一个 iPad。在最初的设想里它就不单单是当平板做笔记看书用的,而是要充当电脑的编程功能,来在必要的时刻随拿随走,替代我这十几斤重、续航还短的游戏本的。 |
@@ -450,9 +354,6 @@ code-server | |||
450 | ```bash | 354 | ```bash |
451 | #!/bin/sh | 355 | #!/bin/sh |
452 | code-server --host "0.0.0.0"> /home/player/useful/codeserver-out.txt 2>&1 & | 356 | code-server --host "0.0.0.0"> /home/player/useful/codeserver-out.txt 2>&1 & |
453 | echo 启动完成!程序已在后台运行。 | ||
454 | echo 如需杀死进程,请以jobs查看后台程序编号并kill %num | ||
455 | echo 或ps查看进程号并kill之。 | ||
456 | ``` | 357 | ``` |
457 | 358 | ||
458 | 其中第一条指令是设置允许外界访问,并指定了由后台运行、所有输出写进`/home/player/useful/codeserver.txt`文件中。后续均为输出信息。 | 359 | 其中第一条指令是设置允许外界访问,并指定了由后台运行、所有输出写进`/home/player/useful/codeserver.txt`文件中。后续均为输出信息。 |
@@ -474,11 +375,8 @@ cert: false | |||
474 | 375 | ||
475 | 杀死的方法: | 376 | 杀死的方法: |
476 | 377 | ||
477 | ```bash | 378 | ```plaintext |
478 | # 运行指令 | 379 | $ ps -ef | grep code-server |
479 | ps -ef | grep code-server | ||
480 | |||
481 | # 输出结果如下 | ||
482 | player 2621 1 0 May29 ? 00:00:00 /usr/lib/code-server/lib/node /usr/lib/code-server --host 0.0.0.0 | 380 | player 2621 1 0 May29 ? 00:00:00 /usr/lib/code-server/lib/node /usr/lib/code-server --host 0.0.0.0 |
483 | player 2646 2621 0 May29 ? 00:03:31 /usr/lib/code-server/lib/node /usr/lib/code-server/out/node/entry | 381 | player 2646 2621 0 May29 ? 00:03:31 /usr/lib/code-server/lib/node /usr/lib/code-server/out/node/entry |
484 | player 3079 2646 0 May29 ? 00:00:57 /usr/lib/code-server/lib/node /usr/lib/code-server/lib/vscode/out/bootstrap-fork --type=ptyHost --logsPath /home/player/.local/share/code-server/logs/20230529T211752 | 382 | player 3079 2646 0 May29 ? 00:00:57 /usr/lib/code-server/lib/node /usr/lib/code-server/lib/vscode/out/bootstrap-fork --type=ptyHost --logsPath /home/player/.local/share/code-server/logs/20230529T211752 |
@@ -491,10 +389,10 @@ player 465057 464494 0 13:42 ? 00:00:35 /usr/lib/code-server/lib/nod | |||
491 | player 579173 574648 0 16:44 pts/1 00:00:00 grep --color=auto code-server | 389 | player 579173 574648 0 16:44 pts/1 00:00:00 grep --color=auto code-server |
492 | 390 | ||
493 | # 只需要杀死根进程也就是第一个进程就好,下边的数据需要改成当前对应进程号 | 391 | # 只需要杀死根进程也就是第一个进程就好,下边的数据需要改成当前对应进程号 |
494 | kill -9 2621 | 392 | $ kill -9 2621 |
495 | ``` | 393 | ``` |
496 | 394 | ||
497 | 重新启动之后,就可以用`<服务器公网ip>:<指定端口>`访问在线 VSCode 了,初次访问会比较慢,耐心等待吧。 | 395 | 重新启动之后,就可以用`ip:post`访问在线 VSCode 了,初次访问会比较慢,耐心等待吧。 |
498 | 396 | ||
499 | ## 同步设置 | 397 | ## 同步设置 |
500 | 398 | ||
@@ -535,16 +433,16 @@ kill -9 2621 | |||
535 | 433 | ||
536 | ## ICP 备案 | 434 | ## ICP 备案 |
537 | 435 | ||
538 | 众所周知,我国是工人阶级领导的、以工农联盟为基础的、人民民主专政的社会主义国家,一切事关宣传、舆论的东西都要在党的领导下,都要经过审查备案,包括建网站和域名 https 使用。 | 436 | 众所周知,我国是“工人阶级领导的、以工农联盟为基础的、人民民主专政的社会主义”国家,一切事关宣传、舆论的东西都要在党的领导下,都要经过审查备案,包括建网站和域名 https 使用。ICP 备案的原则是:**用谁的服务器,在谁那里备案**。以我为例,我在腾讯云买的域名、做的 DNS 解析,但用的是阿里云的服务器,就要在阿里云做 ICP 备案。 |
539 | |||
540 | ICP 备案的原则是:用谁的服务器,在谁那里备案。即以我而言,我在腾讯云买的域名、做的 DNS 解析,但用的是阿里云的服务器,就要在阿里云做 ICP 备案。 | ||
541 | 437 | ||
542 | 备案流程按按着走就行,只是有几点需要特别注意: | 438 | 备案流程按按着走就行,只是有几点需要特别注意: |
543 | 439 | ||
544 | - 备案的省份需要与户籍所在省一致 | 440 | - 备案的省份需要与户籍所在省一致 |
545 | - 备案网站名称不要太普通,容易和别人重复,会被客服小姐姐电话要求订正 | 441 | - 备案网站名称不要太普通,容易和别人重复,会被客服小姐姐电话要求订正 |
546 | - 网站名称不要出现“阁、楼、店”等一些容易误会为企业商业主体网站的名称,否则也会被退回。 | 442 | - 网站名称不要出现“阁、楼、店”等一些容易误会为企业商业主体网站的名称,否则也会被退回。 |
547 | - 网站用途描述要详细,尽量 30 字以上,绝对不要出现“博客”等词,审查非常非常严格,而且极大可能直接被 ban。即使你真要博客,不能暗度陈仓吗? | 443 | - 网站用途描述要详细,尽量 30 字以上 |
444 | - 用途描述绝对不要出现“博客”等词,审查非常非常严格,而且极大可能直接被 ban。即使你真要博客,不能暗度陈仓吗? | ||
445 | - 就算你要开评论区也不要说提及。**朋友,你也不想你的博客隔三差五被警察同志拿着放大镜看吧?** | ||
548 | 446 | ||
549 | 提交备案之后。如有问题客服会打电话通知,并退回修改;阿里云初审需要大概不到 1 天,工信部短信确认需要 1 天,然后发送到备案省(即户籍所在省)的管局,时长最多 20 天,建议提前搞。我用了 6 天。 | 447 | 提交备案之后。如有问题客服会打电话通知,并退回修改;阿里云初审需要大概不到 1 天,工信部短信确认需要 1 天,然后发送到备案省(即户籍所在省)的管局,时长最多 20 天,建议提前搞。我用了 6 天。 |
550 | 448 | ||
@@ -554,16 +452,18 @@ ICP 备案的原则是:用谁的服务器,在谁那里备案。即以我而 | |||
554 | 452 | ||
555 | 没错,我们需要一个代理,根据访问的网址来对应到指定端口。 | 453 | 没错,我们需要一个代理,根据访问的网址来对应到指定端口。 |
556 | 454 | ||
557 | ### 安装 | 455 | ### 安装、运行与常用操作 |
558 | 456 | ||
559 | ```bash | 457 | ```bash |
560 | sudo apt install nginx | 458 | sudo apt install nginx |
561 | ``` | 459 | # 安装完成后如果服务器重启,nginx会自动启动,手动启动命令如下 |
460 | sudo nginx | ||
562 | 461 | ||
563 | ### 运行 | 462 | # 停止服务 |
463 | sudo nginx -s stop | ||
564 | 464 | ||
565 | ```bash | 465 | # 重新加载配置文件(适用于修改了配置文件但不需要重启的场景) |
566 | nginx | 466 | sudo nginx -s reload |
567 | ``` | 467 | ``` |
568 | 468 | ||
569 | ### SSL 证书 | 469 | ### SSL 证书 |
@@ -576,30 +476,21 @@ nginx | |||
576 | sudo apt install certbot | 476 | sudo apt install certbot |
577 | ``` | 477 | ``` |
578 | 478 | ||
579 | 在申请证书之前,需要停止 nginx 的工作,如果正在运行,按照上边说的方法杀死即可: | 479 | 在申请证书之前,需要停止 nginx 的工作,如果正在运行,按照上边说的方法杀死,而后就可以申请证书辣! |
580 | |||
581 | ```bash | ||
582 | ps -ef | grep nginx | ||
583 | |||
584 | # 杀死显示的第一个进程 | ||
585 | kill -9 <进程号> | ||
586 | |||
587 | # 重启nginx | ||
588 | nginx | ||
589 | ``` | ||
590 | |||
591 | 而后就可以申请证书辣! | ||
592 | 480 | ||
593 | ```bash | 481 | ```bash |
594 | # 申请证书 | 482 | # 申请证书 |
595 | sudo certbot certonly --standalone --email example@qq.com -d code.player.com | 483 | sudo certbot certonly --standalone --email example@qq.com -d code.player.com |
484 | |||
485 | # 除了使用参数方式外,也可以使用交互的方式 | ||
486 | sudo certbot certonly | ||
596 | ``` | 487 | ``` |
597 | 488 | ||
598 | 申请完成后会展示证书的存储路径,记下来,后边会用到。比如: | 489 | 申请完成后会展示证书的存储路径,记下来,后边会用到。比如: |
599 | 490 | ||
600 | SSL 证书 /etc/letsencrypt/live/code.play.com/fullchain.pem; | 491 | SSL 证书 /etc/letsencrypt/live/code.player.com/fullchain.pem; |
601 | 492 | ||
602 | SSL 证书秘钥 /etc/letsencrypt/live/code.play.com/privkey.pem; | 493 | SSL 证书秘钥 /etc/letsencrypt/live/code.player.com/privkey.pem; |
603 | 494 | ||
604 | ### 修改、添加配置 | 495 | ### 修改、添加配置 |
605 | 496 | ||
@@ -610,6 +501,7 @@ Nginx 默认下载目录在`/etc/nginx`,该目录下有个`.conf`文件,但 | |||
610 | ```conf | 501 | ```conf |
611 | server | 502 | server |
612 | { | 503 | { |
504 | # 这段是一个常用的http重定向到https的方法,用了都说好 | ||
613 | listen 80; | 505 | listen 80; |
614 | server_name code.player.com; | 506 | server_name code.player.com; |
615 | return 301 https://$host$request_uri; | 507 | return 301 https://$host$request_uri; |
@@ -619,28 +511,128 @@ server | |||
619 | server_name code.player.com; | 511 | server_name code.player.com; |
620 | listen 443 ssl; | 512 | listen 443 ssl; |
621 | 513 | ||
514 | # 证书路径 | ||
622 | ssl_certificate /etc/letsencrypt/live/code.player.com/fullchain.pem; | 515 | ssl_certificate /etc/letsencrypt/live/code.player.com/fullchain.pem; |
623 | ssl_certificate_key /etc/letsencrypt/live/code.player.com/privkey.pem; | 516 | ssl_certificate_key /etc/letsencrypt/live/code.player.com/privkey.pem; |
624 | 517 | ||
518 | # 这是对于code-server做出的设置 | ||
625 | location / { | 519 | location / { |
626 | proxy_pass http://127.0.0.1:8080;# 这里的端口号要和code-server指定的端口号一致,我是8080 | 520 | proxy_pass http://127.0.0.1:8080; # 这里的端口号要和code-server指定的端口号一致,我是8080 |
627 | proxy_set_header Host $host; | 521 | proxy_set_header Host $host; |
628 | proxy_set_header Upgrade $http_upgrade; | 522 | proxy_set_header Upgrade $http_upgrade; |
629 | proxy_set_header Connection upgrade; | 523 | proxy_set_header Connection upgrade; |
630 | proxy_set_header Accept-Encoding gzip; | 524 | proxy_set_header Accept-Encoding gzip; |
631 | } | 525 | } |
526 | |||
527 | # 如果文件对应的是webroot服务(也就是一个文件夹下存放着指定网址的网页) | ||
528 | # 需要将上边的配置修改为 | ||
529 | # location / { | ||
530 | # root /var/myweb/; # 这里是文件夹路径 | ||
531 | # index index.html; # 这里是每个文件夹下默认打开的文件 | ||
532 | # } | ||
632 | } | 533 | } |
633 | ``` | 534 | ``` |
634 | 535 | ||
635 | 亲测有效。 | ||
636 | |||
637 | 编写完文件之后,需要重新加载配置文件(不必重启 Nginx): | 536 | 编写完文件之后,需要重新加载配置文件(不必重启 Nginx): |
638 | 537 | ||
639 | ```bash | 538 | ```bash |
640 | nginx -s reload | 539 | sudo nginx -s reload |
540 | ``` | ||
541 | |||
542 | 再次提醒:一定记得把域名换成自己的域名、证书路径换成自己的证书路径、端口换成自己在提供服务的应用(如 code-server)中指定的端口! | ||
543 | |||
544 | # 服务器上的文件浏览 | ||
545 | |||
546 | 我们已经学会了设置 nginx 的 webroot,但也有很多情况我们需要能在各种设备上直接地看到服务器上有哪些文件,既能方便文件传输,也能覆盖对大多数只需要看不需要改的场景。 | ||
547 | |||
548 | python3 为我们提供了一个简单的 http 文件浏览服务,默认安装目录为`/usr/lib/python3.x/http/server.py`,一般情况下我们可以直接运行它: | ||
549 | |||
550 | ```bash | ||
551 | python3 -m http.server | ||
552 | ``` | ||
553 | |||
554 | 但是 python 提供的这份文件存在一些问题: | ||
555 | |||
556 | - 响应头没有规定文件编码,让浏览器自己看着办,容易出现乱码 | ||
557 | - 里边选择性忽略了一些文件、文件夹 | ||
558 | |||
559 | 于是我在给定的基础上做了一些小修改,修改版的文件在[这里](https://www.qin-juan-ge-zhu.top/code/linux/httpmyserver.html)。 | ||
560 | |||
561 | # git 远程存储 | ||
562 | |||
563 | 君不闻《西游记》有云: | ||
564 | |||
565 | > 争名夺利几时休,早起迟眠不自由。 | ||
566 | > | ||
567 | > 骑着驴骡思骏马,官居宰相望王侯。 | ||
568 | |||
569 | 完成了代码环境的配置,我就在想,要是有一个自己的远程 git 托管环境该多好?那就干! | ||
570 | |||
571 | 我的详细配置方法写在[这里](https://www.qin-juan-ge-zhu.top/code/linux/gitserver.html)里了。 | ||
572 | |||
573 | # 游戏 | ||
574 | |||
575 | > All work no play makes Jack a doll boy. | ||
576 | |||
577 | 经过了上边的配置,写代码的控制基本完善了,但是服务器怎么能只拿来干活呢?没有游戏怎么行?我们要快乐! | ||
578 | |||
579 | 目前我装了三个游戏: | ||
580 | |||
581 | - 俄罗斯方块,装的是`tint` | ||
582 | - 扫雷,装的是`minebash` | ||
583 | - 贪吃蛇,找不到力(允悲 | ||
584 | |||
585 | 安装方法网上都有,就不多说了。玩得愉快! | ||
586 | |||
587 | # 开机运行脚本 | ||
588 | |||
589 | 经过了以上的设置,我的服务器已经有了许多服务,特别是每个计算机人电脑里必然需要养的一只宠物——小蓝猫 clash。但是每次重启系统之后,这许多服务都需要自己手动拉起的话未免太过麻烦了,所以我们需要一个在系统加载完成后就能自动以 root 身份执行的脚本。经过查询,我找到了处理办法。 | ||
590 | |||
591 | 在 Linux 系统下,开机启动一般使用的是`/etc/rc.local`文件(但也有很多发行版不再使用这种操作方式)。ubuntu20.04 系统已经默认安装了 rc-local.service 服务,但是不知什么原因系统把这个服务给“隐蔽”了,所以如果不做一番操作是无法使用的。 | ||
592 | |||
593 | ```bash | ||
594 | # 以下所有命令需要root身份执行 | ||
595 | |||
596 | cp /usr/lib/systemd/system/rc-local.service /etc/systemd/system/ | ||
597 | # 修改rc-local.service文件,在文件末尾添加以下内容(注意删去前边的注释符): | ||
598 | # [Install] | ||
599 | # WantedBy=multi-user.target | ||
600 | |||
601 | # 创建rc.local文件,带上shebang行 | ||
602 | echo '#!/bin/bash' > /etc/rc.local | ||
603 | # 修改rc.local文件权限 | ||
604 | chmod +x /etc/rc.local | ||
605 | |||
606 | # 上述步骤完成后,注意不要急于写rc.local脚本,我们需要先启动rc-local.service服务 | ||
607 | systemctl start rc-local.service | ||
608 | systemctl enable rc-local.service | ||
609 | # 上述两条命令运行正常时,系统需要重启 | ||
610 | reboot | ||
611 | |||
612 | # 重启完成后,检查rc-local.service服务是否正常运行 | ||
613 | systemctl status rc-local.service | ||
614 | |||
615 | # 一切正常,就可以编写rc.local脚本的内容了 | ||
616 | # 需要注意的是,该脚本以root身份运行,所以不需要sudo | ||
617 | # 如果部分命令需要以其他用户身份运行,可以使用su命令,如: | ||
618 | # su -l <username> -c <要运行的命令> | ||
619 | # 如果要运行的是脚本(不建议,没必要),注意对应参数 | ||
620 | # 文件的最后,可以用exec &> /var/log/rc-local.log将脚本的输出重定向到日志文件中 | ||
621 | ``` | ||
622 | |||
623 | 除此之外,我们可能还需要为所有用户默认开启魔法。这个功能在开机脚本和`/etc/profile`中都无法实现,应当放在`/etc/environment`中。在其中添加以下内容: | ||
624 | |||
625 | ```plaintext | ||
626 | PATH="/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin:/usr/games:/usr/local/games:/snap/bin" | ||
627 | http_proxy=http://127.0.0.1:7890 | ||
628 | https_proxy=http://127.0.0.1:7890 | ||
629 | no_proxy=127.0.0.1,localhost | ||
630 | HTTP_PROXY=http://127.0.0.1:7890 | ||
631 | HTTPS_PROXY=http://127.0.0.1:7890 | ||
632 | NO_PROXY=127.0.0.1,localhost | ||
641 | ``` | 633 | ``` |
642 | 634 | ||
643 | 但在使用过程中,一定记得把域名换成自己的域名、证书路径换成自己的证书路径、端口换成自己在 code-server 中指定的端口! | 635 | 该文件中的所有设置对于所有用户均有效,着实省去了每次开机、切换用户都要手动挂代理的麻烦。 |
644 | 636 | ||
645 | # 碎碎念 | 637 | # 碎碎念 |
646 | 638 | ||
@@ -652,4 +644,4 @@ nginx -s reload | |||
652 | 644 | ||
653 | 这篇文档是对一个月以来搞云服务器的配置的一个小小总结,本来以为没多少东西,结果写了整整一下午到现在,一个 Markdown 文档,手写居然达到了 900 多行、将近 40KB,令我也大吃一惊。 | 645 | 这篇文档是对一个月以来搞云服务器的配置的一个小小总结,本来以为没多少东西,结果写了整整一下午到现在,一个 Markdown 文档,手写居然达到了 900 多行、将近 40KB,令我也大吃一惊。 |
654 | 646 | ||
655 | 也许我之后还会继续鼓捣我的服务器,那时候就继续写下去吧。现在也许该考虑考虑期末考试了(笑。 | 647 | 也许我之后还会继续鼓捣我的服务器,那时候就继续写下去吧。现在该考虑考虑期末考试了(笑。 |
diff --git a/code/linux/gitserver.html b/code/linux/gitserver.html new file mode 100644 index 0000000..359d948 --- /dev/null +++ b/code/linux/gitserver.html | |||
@@ -0,0 +1,364 @@ | |||
1 | <!DOCTYPE html> | ||
2 | <html xmlns="http://www.w3.org/1999/xhtml" lang="" xml:lang=""> | ||
3 | |||
4 | <head> | ||
5 | <meta charset="utf-8" /> | ||
6 | <meta name="generator" content="pandoc" /> | ||
7 | <meta name="viewport" content="width=device-width, initial-scale=1.0, user-scalable=yes" /> | ||
8 | <title>自己的git托管</title> | ||
9 | <link rel="stylesheet" href="https://www.qin-juan-ge-zhu.top/common/CSS/pandoc.css"> | ||
10 | <script type="text/javascript" src="https://www.qin-juan-ge-zhu.top/common/script4code.js"></script> | ||
11 | </head> | ||
12 | |||
13 | <body> | ||
14 | <div class="pandoc"> | ||
15 | <div class="main"> | ||
16 | <p class="title">自己的git托管</p> | ||
17 | <!-- git 远程存储 --> | ||
18 | <blockquote> | ||
19 | <p>争名夺利几时休,早起迟眠不自由。</p> | ||
20 | <p>骑着驴骡思骏马,官居宰相望王侯。</p> | ||
21 | </blockquote> | ||
22 | <p>自打在服务器上完成了代码环境的配置,我就在想,要是有一个自己的远程 git 存储库该多好?</p> | ||
23 | <h1 id="gitlab">gitlab</h1> | ||
24 | <p>我按照网上的教程,我更新了清华镜像源,下载安装 gitlab。孰料一次一次在安装过程卡死,而且情况十分严重,CPU 占用率长期维持在 50%以上,内存更是飙升到 | ||
25 | 90%,不仅安装进行不下去了,甚至一切操作都无法执行,包括<kbd>Ctrl+C</kbd>想要杀死当前进程的请求也会石沉大海。没办法,只得强制重启。</p> | ||
26 | <p>如是反复若干次,最终死心,将 gitlab 彻底从云服务器卸载,“革职为民,永不叙用”。</p> | ||
27 | <p>经过上网查找,原因果然出现在配置上。<strong>gitlab 推荐的最小内存是 4G,但是目前我们只有 | ||
28 | 2G,内存爆满也就理所当然了。</strong>没办法,期限之内不能更改配置,就算能更改,我都穷到薅资本主义羊毛了,还能有钱升级配置?笑死。</p> | ||
29 | <p>就因为这个问题,我们就不干了?这是不行滴,小同志。</p> | ||
30 | <p>不久之后,黄四郎同志发来了一篇阮一峰的博客:<a href="https://www.ruanyifeng.com/blog/2022/10/git-server.html">最简单的 git | ||
31 | 服务器</a>。看起来似乎不错,但是失之简略。幸好,我在 git 官方教程<a href="https://git-scm.com/book/zh/v2">Pro | ||
32 | Git</a>上找到了另一部分,两个拼一拼、试一试,最终成功了。</p> | ||
33 | <h1 id="前置">前置</h1> | ||
34 | <p>首先,我建议为 git 托管创建专门的用户。这样做有以下几个目的:</p> | ||
35 | <ul> | ||
36 | <li>安全 | ||
37 | <ul> | ||
38 | <li>如果使用常用用户,仓库托管在家目录下的某一级目录,容易一个不注意自己把仓库删了</li> | ||
39 | <li>如果使用 root 用户,找一个目录进行管理,则权限过高,也容易被误操作;除此之外,root 用户一般不会设置为 ssh 登录,这就意味着想简单创建一个库,我们需要先 ssh | ||
40 | 登陆,然后切换到 root 用户、进入存放目录/创建,最后退出,整个操作过程较为烦人且无法自动化实现</li> | ||
41 | </ul> | ||
42 | </li> | ||
43 | <li>方便使用。创建专门用户后,我们可以为之配置 ssh 密钥,从而实现免密登陆,而且创建新仓库、clone 仓库都较为简单(因为直接托管在家目录之下,不需要记较长的路径名)</li> | ||
44 | </ul> | ||
45 | <p>当然,创建单独用户的理由不止这么简单,它在我们使用 http 来托管仓库的时候有专门的作用,下文会讨论到。</p> | ||
46 | <p>我为其创建了一个名为<code>git</code>的用户,家目录<code>/home/git</code>,但不授予 sudo 权限。</p> | ||
47 | <h1 id="ssh-服务">ssh 服务</h1> | ||
48 | <p>首先,本方法要求有一个本地 git 存储库,并且已经有 commit。如果是本地新建的仓库,需要先有一个 commit,然后继续操作。这个 commit | ||
49 | 可以用编写<code>.gitignore</code>或者<code>push.sh</code>等不甚重要的文件来凑数。</p> | ||
50 | <h2 id="远程仓库">远程仓库</h2> | ||
51 | <pre><code># 创建远程仓库,这里也叫test吧 | ||
52 | ssh git@127.0.0.1 git init --bare test.git</code></pre> | ||
53 | <p>注意,本处指明的 git 为远程用户名,127.0.0.1 代表云服务器的公网 ip,test.git | ||
54 | 为远程仓库名。使用的时候都需要换成自己的。如果仓库不想直接存储在用户目录下,需要指出其存储路径,如<code>code/fuck/test.git</code></p> | ||
55 | <h2 id="本地与远程连接">本地与远程连接</h2> | ||
56 | <p>本地和远程都有了,下一步就是建立联系了。</p> | ||
57 | <pre><code># 本地添加远程库信息 | ||
58 | git remote add origin git@127.0.0.1:test.git | ||
59 | |||
60 | # 此时尚不能直接推送,因为并未指定上游对应分支,需要指定 | ||
61 | git push --set-upstream origin master | ||
62 | |||
63 | # 之后就可以直接推送了 | ||
64 | bash push.sh</code></pre> | ||
65 | <h2 id="自动化">自动化</h2> | ||
66 | <p>整体操作过程可以说比较简单了。既然如此,<strong>脚本,启动!</strong></p> | ||
67 | <pre><code>#!/bin/bash | ||
68 | |||
69 | read -p "Local repo name: " local_name | ||
70 | read -p "Remote repo name: " remote_name | ||
71 | |||
72 | # 远程仓库创建 | ||
73 | tmp="ssh aliyun-git git init --bare $remote_name.git" | ||
74 | eval "$tmp" | ||
75 | |||
76 | # 本地仓库创建 | ||
77 | mkdir "$local_name" | ||
78 | cd "$local_name" | ||
79 | git init | ||
80 | |||
81 | # 本地仓库初始化 | ||
82 | # 编写.gitignore | ||
83 | cat > .gitignore << EOF | ||
84 | *.sh | ||
85 | *.bat | ||
86 | *.exe | ||
87 | *.[oa] | ||
88 | *.pyc | ||
89 | __pycache__ | ||
90 | *.vscode | ||
91 | *.swp | ||
92 | EOF | ||
93 | |||
94 | # 编写push.sh | ||
95 | cat > push.sh << EOF | ||
96 | git add . | ||
97 | git commit | ||
98 | git push | ||
99 | EOF | ||
100 | chmod +x push.sh | ||
101 | |||
102 | # 提交初始化commit | ||
103 | git add . | ||
104 | git commit -m "Initial commit" | ||
105 | tmp="git remote add origin aliyun-git:$remote_name.git" | ||
106 | eval "$tmp" | ||
107 | git push --set-upstream origin master | ||
108 | |||
109 | echo "Success!"</code></pre> | ||
110 | <p><strong>Attention please:</strong></p> | ||
111 | <ul> | ||
112 | <li>本地仓库名和远程仓库名不要加.git 后缀,程序会自动添加</li> | ||
113 | <li>如果要把这个脚本改写为其他语言的脚本的话,在转换本地当前工作目录(进入本地新创建的仓库)的时候,<strong>一定要使用语言自带的调整当前工作目录的函数,不要调用系统命令</strong>(如写为 | ||
114 | C | ||
115 | 语言程序的时候,应当是调用库函数<code>chdir()</code>而不是选择<code>system("cd <目录名>")</code>),否则编译器/解释器自动在这里作多线程执行,创建一个新的线程,运行一个终端,进入了该目录,然后没有后续,线程挂了;另一个线程在原来的代码上继续执行,但根本没进入目录,也就是执行操作的位置不对,最终出现错误。 | ||
116 | </li> | ||
117 | </ul> | ||
118 | <h1 id="https服务">http(s)服务</h1> | ||
119 | <p>到这里似乎已经万事俱备了,但在服务器迁移的过程中,我发现了问题:使用 ssh 服务,就意味着每个仓库记录的远程库都是其 ip | ||
120 | 地址(或者自己在<code>~/.ssh/config</code>里规定的服务器别名),当服务器迁移后,需要手动进入一个个本地仓库修改远程库地址,非常麻烦。相比之下,如果使用像 github 一样的 | ||
121 | http(s)服务,我们就只需要做好远程库迁移、dns 信息修改和服务器设置即可,本地仓库完全不需要改动。</p> | ||
122 | <p>于是,我发现了一篇<a | ||
123 | href="https://www.aneasystone.com/archives/2018/12/build-your-own-git-server.html">教程</a>,照葫芦画瓢起来。 | ||
124 | </p> | ||
125 | <p>需要注意的是,<strong>上面的 ssh 服务不一定需要专门创建用户,但 http(s)服务这里我推荐新建用户</strong>。</p> | ||
126 | <h2 id="配套软件安装">配套软件安装</h2> | ||
127 | <p>首先,我们需要安装<code>git</code>和<code>nginx</code>,这里不再赘述。除此之外,我们会用到以下软件:</p> | ||
128 | <ul> | ||
129 | <li><code>git-http-backend</code>:git 官方提供的 git 的 http(s) 服务后端。该软件包含在<code>git-core</code>中,一般会在安装 git | ||
130 | 时就默认装了,可以通过<code>which git-http-backend</code>查看是否安装,没有的话需要安装<code>git-core</code>。默认安装位置在<code>/usr/lib/git-core/git-http-backend</code>,需要查看自己的路径,后边会用。 | ||
131 | </li> | ||
132 | <li><code>fcgiwrap</code>:一个用于将 FastCGI 转换为 HTTP 协议的工具。</li> | ||
133 | <li><code>htpasswd</code>:用于创建和更新基于文本的认证文件,用于存储用户名和密码。它是<code>apache2-utils</code>的一部分,可以通过<code>which htpasswd</code>查看是否安装,没有的话需要安装<code>apache2-utils</code>。 | ||
134 | </li> | ||
135 | </ul> | ||
136 | <h2 id="nginx-配置与说明">nginx 配置与说明</h2> | ||
137 | <p>最初的设置是这样的:</p> | ||
138 | <pre><code>server | ||
139 | { | ||
140 | # 这里用于将 http 请求重定向到 https,是一种常用的方式 | ||
141 | listen 80; | ||
142 | server_name git.player.com; | ||
143 | return 301 https://$host$request_uri; | ||
144 | } | ||
145 | server | ||
146 | { | ||
147 | server_name git.player.com; | ||
148 | listen 443 ssl; | ||
149 | |||
150 | ssl_certificate /etc/letsencrypt/live/git.player.com/fullchain.pem; | ||
151 | ssl_certificate_key /etc/letsencrypt/live/git.player.com/privkey.pem; | ||
152 | |||
153 | location / { | ||
154 | include fastcgi_params; | ||
155 | # git-http-backend的路径,每次请求都会转发到这个脚本 | ||
156 | fastcgi_param SCRIPT_FILENAME /usr/lib/git-core/git-http-backend; | ||
157 | # git-http-backend默认只能访问目录下有git-daemon-export-ok的仓库 | ||
158 | # 这里设置为可以访问所有仓库 | ||
159 | fastcgi_param GIT_HTTP_EXPORT_ALL ""; | ||
160 | # git仓库根目录 | ||
161 | fastcgi_param GIT_PROJECT_ROOT /home/git/; | ||
162 | fastcgi_param PATH_INFO $uri; | ||
163 | # 如果有认证,传递给cgi脚本git-http-backend | ||
164 | fastcgi_param REMOTE_USER $remote_user; | ||
165 | fastcgi_pass unix:/var/run/fcgiwrap.socket; | ||
166 | } | ||
167 | }</code></pre> | ||
168 | <p>nginx 重新加载配置之后,我们就可以正常<code>git clone https://git.player.com/test.git</code>了。但当我们 push 时,会出现 403 错误:</p> | ||
169 | <pre><code>$ git push | ||
170 | fatal: unable to access 'https://git.player.com/test.git/': The requested URL returned error: 403</code></pre> | ||
171 | <p>为了解决这个错误,我们可以在 <code>git-http-backend</code> 的官网文档 上找到这样的一段描述:</p> | ||
172 | <blockquote> | ||
173 | <p>By default, only the upload-pack service is enabled, which serves git fetch-pack and git ls-remote | ||
174 | clients, which are invoked from git fetch, git pull, and git clone. If the client is authenticated, | ||
175 | the receive-pack service is enabled, which serves git send-pack clients, which is invoked from git | ||
176 | push.</p> | ||
177 | </blockquote> | ||
178 | <p>第一次读这段话可能会有些不知所云,这是因为我们对这里提到的 <code>upload-pack/fetch-pack/receive-pack/send-pack</code> | ||
179 | 这几个概念还没有什么认识。但是我们大抵可以猜出来,默认情况下,只有认证的用户才可以 push 代码。</p> | ||
180 | <p>我们可以在仓库中执行 <code>git config http.receivepack true</code> 来开启 push 权限,但是这样的话,所有人都可以 push | ||
181 | 代码了,这显然不是我们想要的。我们可以通过 <code>git config http.receivepack false</code> 来关闭 push 权限,这样的话,所有人都不能 push | ||
182 | 代码了,这也不是我们想要的。那么,我们应该怎么做呢?更好的做法是这样的:</p> | ||
183 | <pre><code>$HTTP["querystring"] =~ "service=git-receive-pack" { | ||
184 | include "git-auth.conf" | ||
185 | } | ||
186 | $HTTP["url"] =~ "^/git/.*/git-receive-pack$" { | ||
187 | include "git-auth.conf" | ||
188 | }</code></pre> | ||
189 | <p>看上去挺简单,但是想要理解为什么这样配置,就必须了解下 Git 的内部原理。正如上面 git-http-backend 文档上的那段描述,当 Git 客户端执行 | ||
190 | <code>git fetch/git pull/git clone</code>时,会调用 <code>upload-pack</code> 服务,当执行 <code>git push</code> | ||
191 | 时,会调用 <code>receive-pack</code> 服务。我们可以查看 nginx 的访问日志,目录在<code>/var/log/nginx/access.log</code>: | ||
192 | </p> | ||
193 | <p>执行 git clone:</p> | ||
194 | <pre><code>[27/Nov/2018:22:18:00] "GET /test.git/info/refs?service=git-upload-pack HTTP/1.1" 200 363 "-" "git/1.9.1" | ||
195 | [27/Nov/2018:22:18:00] "POST /test.git/git-upload-pack HTTP/1.1" 200 306 "-" "git/1.9.1"</code></pre> | ||
196 | <p>执行 git pull:</p> | ||
197 | <pre><code>[27/Nov/2018:22:20:25] "GET /test.git/info/refs?service=git-upload-pack HTTP/1.1" 200 363 "-" "git/1.9.1" | ||
198 | [27/Nov/2018:22:20:25] "POST /test.git/git-upload-pack HTTP/1.1" 200 551 "-" "git/1.9.1"</code></pre> | ||
199 | <p>执行 git push:</p> | ||
200 | <pre><code>[27/Nov/2018:22:19:33] "GET /test.git/info/refs?service=git-receive-pack HTTP/1.1" 401 204 "-" "git/1.9.1" | ||
201 | admin [27/Nov/2018:22:19:33] "GET /test.git/info/refs?service=git-receive-pack HTTP/1.1" 200 193 "-" "git/1.9.1" | ||
202 | admin [27/Nov/2018:22:19:33] "POST /test.git/git-receive-pack HTTP/1.1" 200 63 "-" "git/1.9.1"</code></pre> | ||
203 | <p>可以看到执行 clone 和 pull 请求的接口是一样的,先请求 <code>/info/refs?service=git-upload-pack</code>,然后再请求 | ||
204 | <code>/git-upload-pack</code>;而 push 是先请求 <code>/info/refs?service=git-receive-pack</code>,然后再请求 | ||
205 | <code>/git-receive-pack</code>,所以在上面的的配置中我们看到了两条记录,如果要对 push 做访问控制,那么对这两个请求都要限制。关于 Git 传输的原理可以参考 《Pro | ||
206 | Git》的 <a | ||
207 | href="https://git-scm.com/book/zh/v2/Git-%E5%86%85%E9%83%A8%E5%8E%9F%E7%90%86-%E4%BC%A0%E8%BE%93%E5%8D%8F%E8%AE%AE">Git | ||
208 | 内部原理 - 传输协议</a> 这一节。 | ||
209 | </p> | ||
210 | <p>于是,我们对 nginx 的配置文件进行修改:</p> | ||
211 | <pre><code>server | ||
212 | { | ||
213 | listen 80; | ||
214 | server_name git.qin-juan-ge-zhu.top; | ||
215 | return 301 https://$host$request_uri; | ||
216 | } | ||
217 | server | ||
218 | { | ||
219 | server_name git.qin-juan-ge-zhu.top; | ||
220 | listen 443 ssl; | ||
221 | |||
222 | ssl_certificate /etc/letsencrypt/live/git.qin-juan-ge-zhu.top/fullchain.pem; | ||
223 | ssl_certificate_key /etc/letsencrypt/live/git.qin-juan-ge-zhu.top/privkey.pem; | ||
224 | |||
225 | location @auth { | ||
226 | auth_basic "Git Server"; | ||
227 | # 生成的认证文件,下文专门说明 | ||
228 | auth_basic_user_file /etc/nginx/conf.d/git.htpasswd; | ||
229 | |||
230 | include fastcgi_params; | ||
231 | # git-http-backend的路径,每次请求都会转发到这个脚本 | ||
232 | fastcgi_param SCRIPT_FILENAME /usr/lib/git-core/git-http-backend; | ||
233 | # git-http-backend默认只能访问目录下有git-daemon-export-ok的仓库 | ||
234 | # 这里设置为可以访问所有仓库 | ||
235 | fastcgi_param GIT_HTTP_EXPORT_ALL ""; | ||
236 | # git仓库根目录 | ||
237 | fastcgi_param GIT_PROJECT_ROOT /home/git/; | ||
238 | fastcgi_param PATH_INFO $uri; | ||
239 | # 如果有认证,传递给cgi脚本git-http-backend | ||
240 | fastcgi_param REMOTE_USER $remote_user; | ||
241 | fastcgi_pass unix:/var/run/fcgiwrap.socket; | ||
242 | } | ||
243 | |||
244 | location / { | ||
245 | error_page 418 = @auth; | ||
246 | if ( $query_string = "service=git-receive-pack" ) { | ||
247 | return 418; | ||
248 | } | ||
249 | if ( $uri ~ "git-receive-pack$" ) { | ||
250 | return 418; | ||
251 | } | ||
252 | include fastcgi_params; | ||
253 | |||
254 | fastcgi_param SCRIPT_FILENAME /usr/lib/git-core/git-http-backend; | ||
255 | fastcgi_param GIT_HTTP_EXPORT_ALL ""; | ||
256 | fastcgi_param GIT_PROJECT_ROOT /home/git/; | ||
257 | fastcgi_param PATH_INFO $uri; | ||
258 | fastcgi_param REMOTE_USER $remote_user; | ||
259 | fastcgi_pass unix:/var/run/fcgiwrap.socket; | ||
260 | } | ||
261 | }</code></pre> | ||
262 | <p>这里我们可以看到,我们对<code>/git/.*/git-receive-pack</code>和<code>/info/refs?service=git-receive-pack</code>进行了限制,如果没有认证,就会返回 | ||
263 | 418 错误,然后 nginx 会将其转发到<code>@auth</code>这个 location,这个 location | ||
264 | 会对请求进行认证,如果认证成功,就会将请求转发到<code>fastcgi_pass</code>指定的脚本,也就是<code>git-http-backend</code>,这样就可以正常 push 了。 | ||
265 | </p> | ||
266 | <blockquote> | ||
267 | <p>为什么不直接在<code>/git/.*/git-receive-pack</code>和<code>/info/refs?service=git-receive-pack</code>这两个 | ||
268 | location 中进行认证呢?如果这样做的话,会导致 clone 和 pull 也需要认证,那多不合适了~</p> | ||
269 | <p>418 是什么,http 协议没见过这个啊?这是 http 协议中的一个状态码,叫做 <strong>I’m a | ||
270 | teapot(我是一个茶壶)</strong>,这个状态码是作为一个愚人节玩笑而创建的,但是它确实存在,可以参考 <a | ||
271 | href="https://tools.ietf.org/html/rfc2324">RFC 2324</a>。我们这里用它来做认证标识,以免和别的状态码冲突。当然你也可以自己规定状态码。 | ||
272 | </p> | ||
273 | </blockquote> | ||
274 | <h2 id="认证文件">认证文件</h2> | ||
275 | <p>nginx 的配置中我们用到了认证文件,它的生成是这样的:</p> | ||
276 | <pre><code># 创建认证文件并添加第一个用户 | ||
277 | htpasswd -cd <文件名> <用户名> <密码> | ||
278 | # 在已有的文件中继续添加用户 | ||
279 | htpasswd -d <文件名> <用户名> <密码></code></pre> | ||
280 | <p>到了这里,我们重启 nginx 然后进行测试,也许就可以正常使用了……吗?</p> | ||
281 | <h2 id="问题">问题</h2> | ||
282 | <p>我在测试的时候发现了一个问题:</p> | ||
283 | <pre><code>$ git clone https://git.player.com/test.git | ||
284 | Cloning into 'test'... | ||
285 | fatal: repository 'https://git.player.com/test.git/' not found</code></pre> | ||
286 | <p>看起来很匪夷所思:我们明明把仓库放进了指定目录啊,nginx 也已经配置好了,为什么就是找不到呢?我在网上找了很久也没找到答案。最后自己发现了问题所在:<strong>仓库文件夹权限不对</strong>。 | ||
287 | </p> | ||
288 | <h3 id="linux-权限描述">Linux 权限描述</h3> | ||
289 | <p>当我们使用<code>ls -l</code>命令查看文件夹时,会看到类似这样的输出:</p> | ||
290 | <pre><code>drwxr-xr-x 2 git git 4096 Nov 27 22:19 test.git</code></pre> | ||
291 | <p>其中前 10 个字符就是用来描述该目录权限的。</p> | ||
292 | <ul> | ||
293 | <li>第一位字符代表文件类型 | ||
294 | <ul> | ||
295 | <li><code>d</code>表示目录</li> | ||
296 | <li><code>-</code>表示文件</li> | ||
297 | <li><code>l</code>表示链接文件</li> | ||
298 | <li><code>b</code>表示块设备文件</li> | ||
299 | <li><code>c</code>表示字符设备文件</li> | ||
300 | <li><code>p</code>表示管道文件</li> | ||
301 | <li><code>s</code>表示套接字文件。</li> | ||
302 | </ul> | ||
303 | </li> | ||
304 | <li>接下来 9 个字符每 3 个一组,分别表示文件所有者、文件所有者所在组、其他用户对该文件的权限。 | ||
305 | <ul> | ||
306 | <li><code>r</code>表示可读</li> | ||
307 | <li><code>w</code>表示可写</li> | ||
308 | <li><code>x</code>表示可执行</li> | ||
309 | <li><code>-</code>表示无对应权限</li> | ||
310 | </ul> | ||
311 | </li> | ||
312 | </ul> | ||
313 | <p>其中,读权限和写权限时不言自明的,可是<strong>当我们在讨论文件夹的执行权限的时候,我们究竟在说什么?</strong>其实,这里的执行权限指的是<strong>进入该文件夹的权限</strong>。如果一个文件夹没有执行权限,那么我们就无法进入该文件夹,也就无法访问该文件夹下的文件,即使有对应权限。 | ||
314 | </p> | ||
315 | <p>回到这个问题,我们可以看到,我们仓库存放的文件夹的权限很有可能是没有对应读写或进入权限。所以修改办法也很简单,改权限。</p> | ||
316 | <h3 id="解决方案">解决方案</h3> | ||
317 | <p>修改权限的方法有以下几种:</p> | ||
318 | <ul> | ||
319 | <li>直接把对应库及其下的所有文件/文件夹对所有其他用户赋予读写执行权限(权限使用<code>chmod -R 777 test.git</code>)。缺点很明显: | ||
320 | <ul> | ||
321 | <li>不安全,任何人都可以访问该文件夹下的文件,尤其是 root 和 nginx 用户,前者权限太高,一旦出错足以破坏一切,后者容易导致危险(如强制停止或内存泄漏时)</li> | ||
322 | <li>不方便。每次创建库都需要手动修改仓库及文件的权限,十分不便</li> | ||
323 | </ul> | ||
324 | </li> | ||
325 | <li>不为 git 仓库设立单独用户,直接由原有用户或 root 用户接管一切。缺点: | ||
326 | <ul> | ||
327 | <li>如果使用原有用户,仓库会和其他文件混在一起,不方便管理,且容易误操作</li> | ||
328 | <li>如果使用 root 用户,权限太高,而且一般 root 不会设置为 ssh 登陆,这就意味着想简单创建一个库操作比较复杂</li> | ||
329 | </ul> | ||
330 | </li> | ||
331 | <li>设立单独用户,将 nginx 的工作用户(一般为 www-data 或 | ||
332 | root,在<code>/etc/nginx/nginx.conf</code>文件第一行规定)添加到该用户所在组。这种方式避免了文件混杂不方便管理的问题,也一定程度上避免了权限过高问题(注意,权限问题依然存在),可以说是一个比较好的解决办法。 | ||
333 | </li> | ||
334 | </ul> | ||
335 | <p><strong>这也就是上文我推荐为 git 仓库托管设立单独用户的原因。</strong>因而,最后我选择了第三种方案。具体操作如下:</p> | ||
336 | <pre><code>usermod -aG git root | ||
337 | usermod -aG git www-data</code></pre> | ||
338 | <p>此时再进行测试,应该就可以正常使用了。</p> | ||
339 | <h2 id="其他存在的问题">其他存在的问题</h2> | ||
340 | <p>除了上述问题已经解决之外,还有一些问题依然存在:</p> | ||
341 | <ul> | ||
342 | <li>不管是 nginx 还是 git,在使用 http 上传的时候都会有一定的缓冲区限制,如果上传文件过大或累计多个 commit 才上传,很可能被拒收导致上传失败。这个问题在 github | ||
343 | 上也存在,但是 github 的缓冲区限制比较大,一般不会出现这个问题。但是我们的服务器配置比较低,所以这个问题就比较严重了。解决办法是在 nginx | ||
344 | 的配置文件中添加<code>client_max_body_size 100m;</code>,这样就可以将缓冲区限制扩大到 100M,一般来说足够了。</li> | ||
345 | <li>即使解决了缓冲区大小,偶尔也会被拒收,原因尚未查清</li> | ||
346 | <li>尚未能通过访问特定链接来实现 git 仓库的创建。据说是用 nginx 调用脚本,但暂时没弄出来</li> | ||
347 | <li>没有一个比较好用且功能较为完备的图形化界面(就像 github 那样)。 | ||
348 | <ul> | ||
349 | <li>GitList 的界面看起来不错,而且能展示源码、clone 链接之类的,整体非常像 github 的界面,可惜使用的是我不会的 php 语言,而且没有找到详细一些的安装使用教程 | ||
350 | </li> | ||
351 | <li>cgit 是一个用纯 C 语言开发的一个 git 裸库展示,虽然界面看起来比较古早,但功能也很不错,能展示源码、自由切换分支、方便地查看提交历史(diss 一下 | ||
352 | github,github 查看提交历史看起来真的很不方便很不直观)。美中不足的是<strong>无法在界面上提供 clone | ||
353 | 和源码下载功能</strong>。不过毕竟是个开源软件,而且是我比较熟悉的 C,等有时间有能力了看看自己能不能实现这个功能吧。</li> | ||
354 | </ul> | ||
355 | </li> | ||
356 | </ul> | ||
357 | <p>这篇博客前前后后有二十多天了,该结束了。闲言少叙,看电视去也~</p> | ||
358 | <p class="time">2024.1.3</p> | ||
359 | <script src="https://www.qin-juan-ge-zhu.top/common/js/comment.js"></script> | ||
360 | </div> | ||
361 | </div> | ||
362 | </body> | ||
363 | |||
364 | </html> \ No newline at end of file | ||
diff --git a/code/linux/gitserver.md b/code/linux/gitserver.md new file mode 100644 index 0000000..d0ed6d9 --- /dev/null +++ b/code/linux/gitserver.md | |||
@@ -0,0 +1,368 @@ | |||
1 | <!-- git 远程存储 --> | ||
2 | |||
3 | > 争名夺利几时休,早起迟眠不自由。 | ||
4 | > | ||
5 | > 骑着驴骡思骏马,官居宰相望王侯。 | ||
6 | |||
7 | 自打在服务器上完成了代码环境的配置,我就在想,要是有一个自己的远程 git 存储库该多好? | ||
8 | |||
9 | # gitlab | ||
10 | |||
11 | 我按照网上的教程,我更新了清华镜像源,下载安装 gitlab。孰料一次一次在安装过程卡死,而且情况十分严重,CPU | ||
12 | 占用率长期维持在 50%以上,内存更是飙升到 90%,不仅安装进行不下去了,甚至一切操作都无法执行,包括<kbd>Ctrl+C</kbd>想要杀死当前进程的请求也会石沉大海。没办法,只得强制重启。 | ||
13 | |||
14 | 如是反复若干次,最终死心,将 gitlab 彻底从云服务器卸载,"革职为民,永不叙用"。 | ||
15 | |||
16 | 经过上网查找,原因果然出现在配置上。**gitlab 推荐的最小内存是 4G,但是目前我们只有 2G,内存爆满也就理所当然了。**没办法,期限之内不能更改配置,就算能更改,我都穷到薅资本主义羊毛了,还能有钱升级配置?笑死。 | ||
17 | |||
18 | 就因为这个问题,我们就不干了?这是不行滴,小同志。 | ||
19 | |||
20 | 不久之后,黄四郎同志发来了一篇阮一峰的博客:[最简单的 git 服务器](https://www.ruanyifeng.com/blog/2022/10/git-server.html)。看起来似乎不错,但是失之简略。幸好,我在 git 官方教程[Pro Git](https://git-scm.com/book/zh/v2)上找到了另一部分,两个拼一拼、试一试,最终成功了。 | ||
21 | |||
22 | # 前置 | ||
23 | |||
24 | 首先,我建议为 git 托管创建专门的用户。这样做有以下几个目的: | ||
25 | |||
26 | - 安全 | ||
27 | - 如果使用常用用户,仓库托管在家目录下的某一级目录,容易一个不注意自己把仓库删了 | ||
28 | - 如果使用 root 用户,找一个目录进行管理,则权限过高,也容易被误操作;除此之外,root 用户一般不会设置为 ssh 登录,这就意味着想简单创建一个库,我们需要先 ssh 登陆,然后切换到 root 用户、进入存放目录/创建,最后退出,整个操作过程较为烦人且无法自动化实现 | ||
29 | - 方便使用。创建专门用户后,我们可以为之配置 ssh 密钥,从而实现免密登陆,而且创建新仓库、clone 仓库都较为简单(因为直接托管在家目录之下,不需要记较长的路径名) | ||
30 | |||
31 | 当然,创建单独用户的理由不止这么简单,它在我们使用 http 来托管仓库的时候有专门的作用,下文会讨论到。 | ||
32 | |||
33 | 我为其创建了一个名为`git`的用户,家目录`/home/git`,但不授予 sudo 权限。 | ||
34 | |||
35 | # ssh 服务 | ||
36 | |||
37 | 首先,本方法要求有一个本地 git 存储库,并且已经有 commit。如果是本地新建的仓库,需要先有一个 commit,然后继续操作。这个 commit 可以用编写`.gitignore`或者`push.sh`等不甚重要的文件来凑数。 | ||
38 | |||
39 | ## 远程仓库 | ||
40 | |||
41 | ```bash | ||
42 | # 创建远程仓库,这里也叫test吧 | ||
43 | ssh git@127.0.0.1 git init --bare test.git | ||
44 | ``` | ||
45 | |||
46 | 注意,本处指明的 git 为远程用户名,127.0.0.1 代表云服务器的公网 ip,test.git 为远程仓库名。使用的时候都需要换成自己的。如果仓库不想直接存储在用户目录下,需要指出其存储路径,如`code/fuck/test.git` | ||
47 | |||
48 | ## 本地与远程连接 | ||
49 | |||
50 | 本地和远程都有了,下一步就是建立联系了。 | ||
51 | |||
52 | ```bash | ||
53 | # 本地添加远程库信息 | ||
54 | git remote add origin git@127.0.0.1:test.git | ||
55 | |||
56 | # 此时尚不能直接推送,因为并未指定上游对应分支,需要指定 | ||
57 | git push --set-upstream origin master | ||
58 | |||
59 | # 之后就可以直接推送了 | ||
60 | bash push.sh | ||
61 | ``` | ||
62 | |||
63 | ## 自动化 | ||
64 | |||
65 | 整体操作过程可以说比较简单了。既然如此,**脚本,启动!** | ||
66 | |||
67 | ```bash | ||
68 | #!/bin/bash | ||
69 | |||
70 | read -p "Local repo name: " local_name | ||
71 | read -p "Remote repo name: " remote_name | ||
72 | |||
73 | # 远程仓库创建 | ||
74 | tmp="ssh aliyun-git git init --bare $remote_name.git" | ||
75 | eval "$tmp" | ||
76 | |||
77 | # 本地仓库创建 | ||
78 | mkdir "$local_name" | ||
79 | cd "$local_name" | ||
80 | git init | ||
81 | |||
82 | # 本地仓库初始化 | ||
83 | # 编写.gitignore | ||
84 | cat > .gitignore << EOF | ||
85 | *.sh | ||
86 | *.bat | ||
87 | *.exe | ||
88 | *.[oa] | ||
89 | *.pyc | ||
90 | __pycache__ | ||
91 | *.vscode | ||
92 | *.swp | ||
93 | EOF | ||
94 | |||
95 | # 编写push.sh | ||
96 | cat > push.sh << EOF | ||
97 | git add . | ||
98 | git commit | ||
99 | git push | ||
100 | EOF | ||
101 | chmod +x push.sh | ||
102 | |||
103 | # 提交初始化commit | ||
104 | git add . | ||
105 | git commit -m "Initial commit" | ||
106 | tmp="git remote add origin aliyun-git:$remote_name.git" | ||
107 | eval "$tmp" | ||
108 | git push --set-upstream origin master | ||
109 | |||
110 | echo "Success!" | ||
111 | ``` | ||
112 | |||
113 | **Attention please:** | ||
114 | |||
115 | - 本地仓库名和远程仓库名不要加.git 后缀,程序会自动添加 | ||
116 | - 如果要把这个脚本改写为其他语言的脚本的话,在转换本地当前工作目录(进入本地新创建的仓库)的时候,**一定要使用语言自带的调整当前工作目录的函数,不要调用系统命令**(如写为 C 语言程序的时候,应当是调用库函数`chdir()`而不是选择`system("cd <目录名>")`),否则编译器/解释器自动在这里作多线程执行,创建一个新的线程,运行一个终端,进入了该目录,然后没有后续,线程挂了;另一个线程在原来的代码上继续执行,但根本没进入目录,也就是执行操作的位置不对,最终出现错误。 | ||
117 | |||
118 | # http(s)服务 | ||
119 | |||
120 | 到这里似乎已经万事俱备了,但在服务器迁移的过程中,我发现了问题:使用 ssh 服务,就意味着每个仓库记录的远程库都是其 ip 地址(或者自己在`~/.ssh/config`里规定的服务器别名),当服务器迁移后,需要手动进入一个个本地仓库修改远程库地址,非常麻烦。相比之下,如果使用像 github 一样的 http(s)服务,我们就只需要做好远程库迁移、dns 信息修改和服务器设置即可,本地仓库完全不需要改动。 | ||
121 | |||
122 | 于是,我发现了一篇[教程](https://www.aneasystone.com/archives/2018/12/build-your-own-git-server.html),照葫芦画瓢起来。 | ||
123 | |||
124 | 需要注意的是,**上面的 ssh 服务不一定需要专门创建用户,但 http(s)服务这里我推荐新建用户**。 | ||
125 | |||
126 | ## 配套软件安装 | ||
127 | |||
128 | 首先,我们需要安装`git`和`nginx`,这里不再赘述。除此之外,我们会用到以下软件: | ||
129 | |||
130 | - `git-http-backend`:git 官方提供的 git 的 http(s) 服务后端。该软件包含在`git-core`中,一般会在安装 git 时就默认装了,可以通过`which git-http-backend`查看是否安装,没有的话需要安装`git-core`。默认安装位置在`/usr/lib/git-core/git-http-backend`,需要查看自己的路径,后边会用。 | ||
131 | - `fcgiwrap`:一个用于将 FastCGI 转换为 HTTP 协议的工具。 | ||
132 | - `htpasswd`:用于创建和更新基于文本的认证文件,用于存储用户名和密码。它是`apache2-utils`的一部分,可以通过`which htpasswd`查看是否安装,没有的话需要安装`apache2-utils`。 | ||
133 | |||
134 | ## nginx 配置与说明 | ||
135 | |||
136 | 最初的设置是这样的: | ||
137 | |||
138 | ```conf | ||
139 | server | ||
140 | { | ||
141 | # 这里用于将 http 请求重定向到 https,是一种常用的方式 | ||
142 | listen 80; | ||
143 | server_name git.player.com; | ||
144 | return 301 https://$host$request_uri; | ||
145 | } | ||
146 | server | ||
147 | { | ||
148 | server_name git.player.com; | ||
149 | listen 443 ssl; | ||
150 | |||
151 | ssl_certificate /etc/letsencrypt/live/git.player.com/fullchain.pem; | ||
152 | ssl_certificate_key /etc/letsencrypt/live/git.player.com/privkey.pem; | ||
153 | |||
154 | location / { | ||
155 | include fastcgi_params; | ||
156 | # git-http-backend的路径,每次请求都会转发到这个脚本 | ||
157 | fastcgi_param SCRIPT_FILENAME /usr/lib/git-core/git-http-backend; | ||
158 | # git-http-backend默认只能访问目录下有git-daemon-export-ok的仓库 | ||
159 | # 这里设置为可以访问所有仓库 | ||
160 | fastcgi_param GIT_HTTP_EXPORT_ALL ""; | ||
161 | # git仓库根目录 | ||
162 | fastcgi_param GIT_PROJECT_ROOT /home/git/; | ||
163 | fastcgi_param PATH_INFO $uri; | ||
164 | # 如果有认证,传递给cgi脚本git-http-backend | ||
165 | fastcgi_param REMOTE_USER $remote_user; | ||
166 | fastcgi_pass unix:/var/run/fcgiwrap.socket; | ||
167 | } | ||
168 | } | ||
169 | ``` | ||
170 | |||
171 | nginx 重新加载配置之后,我们就可以正常`git clone https://git.player.com/test.git`了。但当我们 push 时,会出现 403 错误: | ||
172 | |||
173 | ```bash | ||
174 | $ git push | ||
175 | fatal: unable to access 'https://git.player.com/test.git/': The requested URL returned error: 403 | ||
176 | ``` | ||
177 | |||
178 | 为了解决这个错误,我们可以在 `git-http-backend` 的官网文档 上找到这样的一段描述: | ||
179 | |||
180 | > By default, only the upload-pack service is enabled, which serves git fetch-pack and git ls-remote clients, which are invoked from git fetch, git pull, and git clone. If the client is authenticated, the receive-pack service is enabled, which serves git send-pack clients, which is invoked from git push. | ||
181 | |||
182 | 第一次读这段话可能会有些不知所云,这是因为我们对这里提到的 `upload-pack/fetch-pack/receive-pack/send-pack` 这几个概念还没有什么认识。但是我们大抵可以猜出来,默认情况下,只有认证的用户才可以 push 代码。 | ||
183 | |||
184 | 我们可以在仓库中执行 `git config http.receivepack true` 来开启 push 权限,但是这样的话,所有人都可以 push 代码了,这显然不是我们想要的。我们可以通过 `git config http.receivepack false` 来关闭 push 权限,这样的话,所有人都不能 push 代码了,这也不是我们想要的。那么,我们应该怎么做呢?更好的做法是这样的: | ||
185 | |||
186 | ```conf | ||
187 | $HTTP["querystring"] =~ "service=git-receive-pack" { | ||
188 | include "git-auth.conf" | ||
189 | } | ||
190 | $HTTP["url"] =~ "^/git/.*/git-receive-pack$" { | ||
191 | include "git-auth.conf" | ||
192 | } | ||
193 | ``` | ||
194 | |||
195 | 看上去挺简单,但是想要理解为什么这样配置,就必须了解下 Git 的内部原理。正如上面 git-http-backend 文档上的那段描述,当 Git 客户端执行 `git fetch/git pull/git clone`时,会调用 `upload-pack` 服务,当执行 `git push` 时,会调用 `receive-pack` 服务。我们可以查看 nginx 的访问日志,目录在`/var/log/nginx/access.log`: | ||
196 | |||
197 | 执行 git clone: | ||
198 | |||
199 | ```log | ||
200 | [27/Nov/2018:22:18:00] "GET /test.git/info/refs?service=git-upload-pack HTTP/1.1" 200 363 "-" "git/1.9.1" | ||
201 | [27/Nov/2018:22:18:00] "POST /test.git/git-upload-pack HTTP/1.1" 200 306 "-" "git/1.9.1" | ||
202 | ``` | ||
203 | |||
204 | 执行 git pull: | ||
205 | |||
206 | ```log | ||
207 | [27/Nov/2018:22:20:25] "GET /test.git/info/refs?service=git-upload-pack HTTP/1.1" 200 363 "-" "git/1.9.1" | ||
208 | [27/Nov/2018:22:20:25] "POST /test.git/git-upload-pack HTTP/1.1" 200 551 "-" "git/1.9.1" | ||
209 | ``` | ||
210 | |||
211 | 执行 git push: | ||
212 | |||
213 | ```log | ||
214 | [27/Nov/2018:22:19:33] "GET /test.git/info/refs?service=git-receive-pack HTTP/1.1" 401 204 "-" "git/1.9.1" | ||
215 | admin [27/Nov/2018:22:19:33] "GET /test.git/info/refs?service=git-receive-pack HTTP/1.1" 200 193 "-" "git/1.9.1" | ||
216 | admin [27/Nov/2018:22:19:33] "POST /test.git/git-receive-pack HTTP/1.1" 200 63 "-" "git/1.9.1" | ||
217 | ``` | ||
218 | |||
219 | 可以看到执行 clone 和 pull 请求的接口是一样的,先请求 `/info/refs?service=git-upload-pack`,然后再请求 `/git-upload-pack`;而 push 是先请求 `/info/refs?service=git-receive-pack`,然后再请求 `/git-receive-pack`,所以在上面的的配置中我们看到了两条记录,如果要对 push 做访问控制,那么对这两个请求都要限制。关于 Git 传输的原理可以参考 《Pro Git》的 [Git 内部原理 - 传输协议](https://git-scm.com/book/zh/v2/Git-%E5%86%85%E9%83%A8%E5%8E%9F%E7%90%86-%E4%BC%A0%E8%BE%93%E5%8D%8F%E8%AE%AE) 这一节。 | ||
220 | |||
221 | 于是,我们对 nginx 的配置文件进行修改: | ||
222 | |||
223 | ```conf | ||
224 | server | ||
225 | { | ||
226 | listen 80; | ||
227 | server_name git.qin-juan-ge-zhu.top; | ||
228 | return 301 https://$host$request_uri; | ||
229 | } | ||
230 | server | ||
231 | { | ||
232 | server_name git.qin-juan-ge-zhu.top; | ||
233 | listen 443 ssl; | ||
234 | |||
235 | ssl_certificate /etc/letsencrypt/live/git.qin-juan-ge-zhu.top/fullchain.pem; | ||
236 | ssl_certificate_key /etc/letsencrypt/live/git.qin-juan-ge-zhu.top/privkey.pem; | ||
237 | |||
238 | location @auth { | ||
239 | auth_basic "Git Server"; | ||
240 | # 生成的认证文件,下文专门说明 | ||
241 | auth_basic_user_file /etc/nginx/conf.d/git.htpasswd; | ||
242 | |||
243 | include fastcgi_params; | ||
244 | # git-http-backend的路径,每次请求都会转发到这个脚本 | ||
245 | fastcgi_param SCRIPT_FILENAME /usr/lib/git-core/git-http-backend; | ||
246 | # git-http-backend默认只能访问目录下有git-daemon-export-ok的仓库 | ||
247 | # 这里设置为可以访问所有仓库 | ||
248 | fastcgi_param GIT_HTTP_EXPORT_ALL ""; | ||
249 | # git仓库根目录 | ||
250 | fastcgi_param GIT_PROJECT_ROOT /home/git/; | ||
251 | fastcgi_param PATH_INFO $uri; | ||
252 | # 如果有认证,传递给cgi脚本git-http-backend | ||
253 | fastcgi_param REMOTE_USER $remote_user; | ||
254 | fastcgi_pass unix:/var/run/fcgiwrap.socket; | ||
255 | } | ||
256 | |||
257 | location / { | ||
258 | error_page 418 = @auth; | ||
259 | if ( $query_string = "service=git-receive-pack" ) { | ||
260 | return 418; | ||
261 | } | ||
262 | if ( $uri ~ "git-receive-pack$" ) { | ||
263 | return 418; | ||
264 | } | ||
265 | include fastcgi_params; | ||
266 | |||
267 | fastcgi_param SCRIPT_FILENAME /usr/lib/git-core/git-http-backend; | ||
268 | fastcgi_param GIT_HTTP_EXPORT_ALL ""; | ||
269 | fastcgi_param GIT_PROJECT_ROOT /home/git/; | ||
270 | fastcgi_param PATH_INFO $uri; | ||
271 | fastcgi_param REMOTE_USER $remote_user; | ||
272 | fastcgi_pass unix:/var/run/fcgiwrap.socket; | ||
273 | } | ||
274 | } | ||
275 | ``` | ||
276 | |||
277 | 这里我们可以看到,我们对`/git/.*/git-receive-pack`和`/info/refs?service=git-receive-pack`进行了限制,如果没有认证,就会返回 418 错误,然后 nginx 会将其转发到`@auth`这个 location,这个 location 会对请求进行认证,如果认证成功,就会将请求转发到`fastcgi_pass`指定的脚本,也就是`git-http-backend`,这样就可以正常 push 了。 | ||
278 | |||
279 | > 为什么不直接在`/git/.*/git-receive-pack`和`/info/refs?service=git-receive-pack`这两个 location 中进行认证呢?如果这样做的话,会导致 clone 和 pull 也需要认证,那多不合适了~ | ||
280 | > | ||
281 | > 418 是什么,http 协议没见过这个啊?这是 http 协议中的一个状态码,叫做 **I'm a teapot(我是一个茶壶)**,这个状态码是作为一个愚人节玩笑而创建的,但是它确实存在,可以参考 [RFC 2324](https://tools.ietf.org/html/rfc2324)。我们这里用它来做认证标识,以免和别的状态码冲突。当然你也可以自己规定状态码。 | ||
282 | |||
283 | ## 认证文件 | ||
284 | |||
285 | nginx 的配置中我们用到了认证文件,它的生成是这样的: | ||
286 | |||
287 | ```bash | ||
288 | # 创建认证文件并添加第一个用户 | ||
289 | htpasswd -cd <文件名> <用户名> <密码> | ||
290 | # 在已有的文件中继续添加用户 | ||
291 | htpasswd -d <文件名> <用户名> <密码> | ||
292 | ``` | ||
293 | |||
294 | 到了这里,我们重启 nginx 然后进行测试,也许就可以正常使用了……吗? | ||
295 | |||
296 | ## 问题 | ||
297 | |||
298 | 我在测试的时候发现了一个问题: | ||
299 | |||
300 | ```bash | ||
301 | $ git clone https://git.player.com/test.git | ||
302 | Cloning into 'test'... | ||
303 | fatal: repository 'https://git.player.com/test.git/' not found | ||
304 | ``` | ||
305 | |||
306 | 看起来很匪夷所思:我们明明把仓库放进了指定目录啊,nginx 也已经配置好了,为什么就是找不到呢?我在网上找了很久也没找到答案。最后自己发现了问题所在:**仓库文件夹权限不对**。 | ||
307 | |||
308 | ### Linux 权限描述 | ||
309 | |||
310 | 当我们使用`ls -l`命令查看文件夹时,会看到类似这样的输出: | ||
311 | |||
312 | ```bash | ||
313 | drwxr-xr-x 2 git git 4096 Nov 27 22:19 test.git | ||
314 | ``` | ||
315 | |||
316 | 其中前 10 个字符就是用来描述该目录权限的。 | ||
317 | |||
318 | - 第一位字符代表文件类型 | ||
319 | - `d`表示目录 | ||
320 | - `-`表示文件 | ||
321 | - `l`表示链接文件 | ||
322 | - `b`表示块设备文件 | ||
323 | - `c`表示字符设备文件 | ||
324 | - `p`表示管道文件 | ||
325 | - `s`表示套接字文件。 | ||
326 | - 接下来 9 个字符每 3 个一组,分别表示文件所有者、文件所有者所在组、其他用户对该文件的权限。 | ||
327 | - `r`表示可读 | ||
328 | - `w`表示可写 | ||
329 | - `x`表示可执行 | ||
330 | - `-`表示无对应权限 | ||
331 | |||
332 | 其中,读权限和写权限时不言自明的,可是**当我们在讨论文件夹的执行权限的时候,我们究竟在说什么?**其实,这里的执行权限指的是**进入该文件夹的权限**。如果一个文件夹没有执行权限,那么我们就无法进入该文件夹,也就无法访问该文件夹下的文件,即使有对应权限。 | ||
333 | |||
334 | 回到这个问题,我们可以看到,我们仓库存放的文件夹的权限很有可能是没有对应读写或进入权限。所以修改办法也很简单,改权限。 | ||
335 | |||
336 | ### 解决方案 | ||
337 | |||
338 | 修改权限的方法有以下几种: | ||
339 | |||
340 | - 直接把对应库及其下的所有文件/文件夹对所有其他用户赋予读写执行权限(权限使用`chmod -R 777 test.git`)。缺点很明显: | ||
341 | - 不安全,任何人都可以访问该文件夹下的文件,尤其是 root 和 nginx 用户,前者权限太高,一旦出错足以破坏一切,后者容易导致危险(如强制停止或内存泄漏时) | ||
342 | - 不方便。每次创建库都需要手动修改仓库及文件的权限,十分不便 | ||
343 | - 不为 git 仓库设立单独用户,直接由原有用户或 root 用户接管一切。缺点: | ||
344 | - 如果使用原有用户,仓库会和其他文件混在一起,不方便管理,且容易误操作 | ||
345 | - 如果使用 root 用户,权限太高,而且一般 root 不会设置为 ssh 登陆,这就意味着想简单创建一个库操作比较复杂 | ||
346 | - 设立单独用户,将 nginx 的工作用户(一般为 www-data 或 root,在`/etc/nginx/nginx.conf`文件第一行规定)添加到该用户所在组。这种方式避免了文件混杂不方便管理的问题,也一定程度上避免了权限过高问题(注意,权限问题依然存在),可以说是一个比较好的解决办法。 | ||
347 | |||
348 | **这也就是上文我推荐为 git 仓库托管设立单独用户的原因。**因而,最后我选择了第三种方案。具体操作如下: | ||
349 | |||
350 | ```bash | ||
351 | usermod -aG git root | ||
352 | usermod -aG git www-data | ||
353 | ``` | ||
354 | |||
355 | 此时再进行测试,应该就可以正常使用了。 | ||
356 | |||
357 | ## 其他存在的问题 | ||
358 | |||
359 | 除了上述问题已经解决之外,还有一些问题依然存在: | ||
360 | |||
361 | - 不管是 nginx 还是 git,在使用 http 上传的时候都会有一定的缓冲区限制,如果上传文件过大或累计多个 commit 才上传,很可能被拒收导致上传失败。这个问题在 github 上也存在,但是 github 的缓冲区限制比较大,一般不会出现这个问题。但是我们的服务器配置比较低,所以这个问题就比较严重了。解决办法是在 nginx 的配置文件中添加`client_max_body_size 100m;`,这样就可以将缓冲区限制扩大到 100M,一般来说足够了。 | ||
362 | - 即使解决了缓冲区大小,偶尔也会被拒收,原因尚未查清 | ||
363 | - 尚未能通过访问特定链接来实现 git 仓库的创建。据说是用 nginx 调用脚本,但暂时没弄出来 | ||
364 | - 没有一个比较好用且功能较为完备的图形化界面(就像 github 那样)。 | ||
365 | - GitList 的界面看起来不错,而且能展示源码、clone 链接之类的,整体非常像 github 的界面,可惜使用的是我不会的 php 语言,而且没有找到详细一些的安装使用教程 | ||
366 | - cgit 是一个用纯 C 语言开发的一个 git 裸库展示,虽然界面看起来比较古早,但功能也很不错,能展示源码、自由切换分支、方便地查看提交历史(diss 一下 github,github 查看提交历史看起来真的很不方便很不直观)。美中不足的是**无法在界面上提供 clone 和源码下载功能**。不过毕竟是个开源软件,而且是我比较熟悉的 C,等有时间有能力了看看自己能不能实现这个功能吧。 | ||
367 | |||
368 | 这篇博客前前后后有二十多天了,该结束了。闲言少叙,看电视去也~ \ No newline at end of file | ||
diff --git a/code/linux/httpmyserver.html b/code/linux/httpmyserver.html new file mode 100644 index 0000000..9cb878e --- /dev/null +++ b/code/linux/httpmyserver.html | |||
@@ -0,0 +1,277 @@ | |||
1 | <!DOCTYPE html> | ||
2 | <html xmlns="http://www.w3.org/1999/xhtml" lang="" xml:lang=""> | ||
3 | |||
4 | <head> | ||
5 | <meta charset="utf-8" /> | ||
6 | <meta name="generator" content="pandoc" /> | ||
7 | <meta name="viewport" content="width=device-width, initial-scale=1.0, user-scalable=yes" /> | ||
8 | <title>http浏览服务器文件</title> | ||
9 | <link rel="stylesheet" href="https://www.qin-juan-ge-zhu.top/common/CSS/pandoc.css"> | ||
10 | <script type="text/javascript" src="https://www.qin-juan-ge-zhu.top/common/script4code.js"></script> | ||
11 | </head> | ||
12 | |||
13 | <body> | ||
14 | <div class="pandoc"> | ||
15 | <div class="main"> | ||
16 | <p class="title">http浏览服务器文件</p> | ||
17 | <!-- python实现服务器文件浏览 --> | ||
18 | <p>有了服务器,我们自然会有这样的需求:<strong>把一些内容传到服务器上,方便其他人访问或自己在其他设备(如手机)上,通过浏览器直接访问</strong>。我们需要它能完成以下功能:</p> | ||
19 | <ul> | ||
20 | <li>能让我们自己点进(或退出)层层文件夹,展示当前文件夹下的所有文件/文件夹</li> | ||
21 | <li>点击文件,能够看到文件内容</li> | ||
22 | </ul> | ||
23 | <p><img | ||
24 | src="" /> | ||
25 | </p> | ||
26 | <p>点击进入文件<code>nginx.txt</code>,我们要求看到文件的内容:</p> | ||
27 | <p><img | ||
28 | src="" /> | ||
29 | </p> | ||
30 | <p>既要看文件内容,又要能浏览不同的文件夹。这个功能如果直接用程序或 nginx 实现似乎有些新手不友好。没事,python3 | ||
31 | 为我们提供了这个功能,脚本为<code>/usr/lib/python3.x/http/server.py</code>。使用时执行以下命令即可:</p> | ||
32 | <pre><code>python3 -m http.server</code></pre> | ||
33 | <p>但 python 提供的脚本有些不尽人意:</p> | ||
34 | <ul> | ||
35 | <li>遇到当前路径有<code>index.html</code>就展示网页,而不是展示当前路径下的文件</li> | ||
36 | <li>无法回退到上层文件夹</li> | ||
37 | <li>无法查看文件详细信息</li> | ||
38 | <li>浏览普通文件时,响应头没有指定文件编码,导致文件显示乱码</li> | ||
39 | <li>无法方便地上传文件</li> | ||
40 | </ul> | ||
41 | <p>处于以上种种原因,我对该程序做了一点点修改。修改后的脚本我选择放在<code>/usr/lib/python3.x/http/myserver.py</code>(和原有<code>server.py</code>放在一起)。代码全文我放在了<a | ||
42 | href="https://file.qin-juan-ge-zhu.top/useful/myserver.py">这里</a>。在这里仅对修改的部分加以说明。</p> | ||
43 | <p>首先,为了完成以下功能,需要引入几个新的模块:</p> | ||
44 | <pre><code>import cgi | ||
45 | import pwd | ||
46 | import grp | ||
47 | import time | ||
48 | import stat | ||
49 | import math</code></pre> | ||
50 | <h1 id="遇到-index.html-时">遇到 index.html 时</h1> | ||
51 | <p>在<code>http.server</code>(源码路径<code>/usr/lib/python3.x/http/server.py</code>)第 710 | ||
52 | 行(<code>def send_head</code>函数)中有这样一段:</p> | ||
53 | <pre><code>for index in "index.html", "index.htm": | ||
54 | index = os.path.join(path, index) | ||
55 | if os.path.isfile(index): | ||
56 | path = index | ||
57 | break</code></pre> | ||
58 | <p>可以看到,该段代码的作用就是当请求的路径是一个文件夹且该文件夹下有<code>index.html</code>或<code>index.htm</code>,则将该文件作为响应内容返回。这一点对于一般的网页服务器来说是不错的,但咱们需要的是浏览服务器文件,而不是展示网页根目录,所以不需要这个功能,该段代码删除即可。 | ||
59 | </p> | ||
60 | <h1 id="浏览文件出现乱码">浏览文件出现乱码</h1> | ||
61 | <p>你可能会问,上边看到的 html 模板不是已经明文规定编码是<code>utf-8</code>了吗,,怎么还会乱码?没错,html 是说明编码了,但我们除了浏览以 html | ||
62 | 形式展示的目录连接之外,还是要看文件内容的,文件里或多或少都会有点中文不是吗?对于文件而言,编码必须由 http 的响应头指出,否则客户端就会根据文件内容猜测其编码然后展示出来,就成了乱码。</p> | ||
63 | <pre><code># http.server, line 763 | ||
64 | # also in the function 'send_head' | ||
65 | # this line and its context are about http response header | ||
66 | self.send_header("Content-type", ctype) | ||
67 | # change it into the below | ||
68 | self.send_header("Content-type", ctype + "; charset=utf-8")</code></pre> | ||
69 | <h1 id="展示所有文件和文件夹">展示所有文件和文件夹</h1> | ||
70 | <p>在<code>server.py</code>第 772 | ||
71 | 行,有一个<code>list_directory</code>函数,是用来展示当前路径下有哪些文件或文件夹的,但是这有个缺陷,就是只能点进下一级文件夹,但无法返回上一级。为此,我们需要做一些修改:</p> | ||
72 | <pre><code>def list_directory(self, path): | ||
73 | try: | ||
74 | list = os.listdir(path) | ||
75 | except OSError: | ||
76 | self.send_error( | ||
77 | HTTPStatus.NOT_FOUND, | ||
78 | "No permission to list directory") | ||
79 | return None | ||
80 | # 添加这一行,使得可以返回上一级和维持本级 | ||
81 | # 以使展示更贴近于Linux的展示 | ||
82 | list.extend([".",".."]) | ||
83 | # 排序方式原来是完全按照字母顺序,不甚方便 | ||
84 | # 修改为文件夹排在文件之前,各自按照字母排序 | ||
85 | # list.sort(key=lambda a: .lower()) | ||
86 | list.sort(key=lambda a: (not os.path.isdir(os.path.join(path,a)),a.lower()))</code></pre> | ||
87 | <p>可能你已经注意到了,主要的展示目录的界面就是这个函数底下的 html 模板。下面我们会经常修改这个模板的。</p> | ||
88 | <h1 id="文件详细信息展示">文件详细信息展示</h1> | ||
89 | <p>有时候我们需要的不仅仅是文件本身,还需要关注文件的详细信息,如用户权限、文件大小、修改日期之类的。这些信息在 Linux | ||
90 | 系统中通常使用命令<code>ls -l</code>或其别名<code>ll</code>来查看。我想要模仿这个命令的输出格式,做一些微调:</p> | ||
91 | <ul> | ||
92 | <li>第一列是文件名,是一个超链接,这个不必说</li> | ||
93 | <li>第二列是目录类型和其权限位</li> | ||
94 | <li>第三列是目录的硬链接数目,文件是 1,目录则是其下的文件和文件夹总数(包括<code>.</code>和<code>..</code>)</li> | ||
95 | <li>第四列、第五列分别是文件的所有者和所属组</li> | ||
96 | <li>第六列文件大小,按照人可读的方式展示(也就是使用适当的单位)</li> | ||
97 | <li>第七、第八列是创建时间和修改时间</li> | ||
98 | </ul> | ||
99 | <p>为了实现以上目的,我们不能继续使用无序列表来展示信息,而是应当改为表格;同时需要注意的是:</p> | ||
100 | <ul> | ||
101 | <li>既然要展示权限位,必然需要使用等宽字体。这里我选择整个 html 均使用等宽字体</li> | ||
102 | <li>为了美观,目录超链接的样式我做了一些修改,主要是去除了其下划线、设置无论是否访问过都显示为同一个颜色</li> | ||
103 | <li>还做了其他一些小的样式修改</li> | ||
104 | </ul> | ||
105 | <pre><code>enc = sys.getfilesystemencoding() | ||
106 | title = f'Directory listing for {displaypath}' | ||
107 | r.append('<!DOCTYPE HTML>') | ||
108 | r.append('<html lang="en">') | ||
109 | r.append('<head>') | ||
110 | r.append(f'<meta charset="{enc}">') | ||
111 | r.append(f'<title>{title}</title>') | ||
112 | r.append('table{border-collapse:separate;}td,th{padding:0 13px;}.size{text-align:right;}body{font-family:"Courier New",monospace;}h1{font-size:30px;margin-bottom:0;}a{text-decoration:none;}a:link{color:#0000EE;}a:visited{color:#0000EE;}a:hover{color:#0000EE;}a:active{color:#0000EE;}</style>') | ||
113 | r.append('</head><body>') | ||
114 | r.append(f'<h1>{title}</h1>') | ||
115 | r.append('<hr>') | ||
116 | r.append('<table><tr><th>Name</th><th>Permissions</th><th>Hard Links</th><th>Owner</th><th>Group</th><th>Size</th><th>Creation Time</th><th>Last Modified</th></tr>') | ||
117 | for name in list: | ||
118 | fullname = os.path.join(path, name) | ||
119 | displayname = linkname = name | ||
120 | # Append / for directories or @ for symbolic links | ||
121 | if os.path.isdir(fullname): | ||
122 | displayname = name + "/" | ||
123 | linkname = name + "/" | ||
124 | if os.path.islink(fullname): | ||
125 | displayname = name + "@" | ||
126 | # Note: a link to a directory displays with @ and links with / | ||
127 | |||
128 | # Add detailed information for each file/directory | ||
129 | stat_info = os.stat(fullname) | ||
130 | mode = stat.filemode(stat_info.st_mode) | ||
131 | hard_links = stat_info.st_nlink | ||
132 | owner = pwd.getpwuid(stat_info.st_uid).pw_name | ||
133 | group = grp.getgrgid(stat_info.st_gid).gr_name | ||
134 | size_bytes = stat_info.st_size | ||
135 | # 文件大小需要可读 | ||
136 | if size_bytes == 0: | ||
137 | size = '0B' | ||
138 | size_name = (' B', 'KB', 'MB', 'GB') | ||
139 | i = int(math.floor(math.log(size_bytes, 1024))) | ||
140 | p = math.pow(1024, i) | ||
141 | s = size_bytes / p | ||
142 | size = '%.2f %s' % (s, size_name[i]) | ||
143 | creation_time = time.strftime('%Y-%m-%d %H:%M:%S', time.localtime(stat_info.st_ctime)) | ||
144 | last_modified = time.strftime('%Y-%m-%d %H:%M:%S', time.localtime(stat_info.st_mtime)) | ||
145 | |||
146 | r.append('<tr><td><a href="%s">%s</a></td><td>%s</td><td>%d</td><td>%s</td><td>%s</td><td class="size">%s</td><td>%s</td><td>%s</td></tr>' | ||
147 | % (urllib.parse.quote(linkname, errors='surrogatepass'), | ||
148 | html.escape(displayname, quote=False), | ||
149 | mode, hard_links, owner, group, size, creation_time, last_modified)) | ||
150 | r.append('</table>\n</body>\n') | ||
151 | r.append(''' | ||
152 | <script> | ||
153 | onload = () => { | ||
154 | document.getElementById('file').addEventListener('change', function() { | ||
155 | var files = this.files; | ||
156 | var fileNames = ''; | ||
157 | for (var i = 0; i < files.length; i++) { | ||
158 | fileNames += files[i].name + '\\n'; | ||
159 | } | ||
160 | confirm('Are you sure you want to upload these files?\\n' + fileNames); | ||
161 | }); | ||
162 | } | ||
163 | </script> | ||
164 | ''') | ||
165 | r.append('</html>\n') | ||
166 | encoded = '\n'.join(r).encode(enc, 'surrogateescape') | ||
167 | f = io.BytesIO() | ||
168 | f.write(encoded) | ||
169 | f.seek(0) | ||
170 | self.send_response(HTTPStatus.OK) | ||
171 | self.send_header("Content-type", "text/html; charset=%s" % enc) | ||
172 | self.send_header("Content-Length", str(len(encoded))) | ||
173 | self.end_headers() | ||
174 | return f</code></pre> | ||
175 | <p>到了这一步,整体的布局美观了不少,可以称得上是赏心悦目了。</p> | ||
176 | <h1 id="文件上传">文件上传</h1> | ||
177 | <p>其实轮起来,文件上传这个功能我本是不需要的,我所有设备都能通过<code>scp</code>将文件上传到服务器上。这一切,都是被微信逼的。</p> | ||
178 | <p>前两天,东方直心前辈想要用微信给我发他最新修改的《毛泽东大传》,一连几次打包发我,我都没收到,只能解释为被微信截胡了。没办法,我又不能教老人用什么太繁杂的方法,因而一连几天搞这个文件上传功能,最终在黄四郎同志和 | ||
179 | gpt、copilot 的帮助下,完成了这个功能。</p> | ||
180 | <p>首先,我们需要一个按钮来供使用者上传文件。为了整体页面协调,我选择将上传表单与 h1 放在同一行,都在分割线上方,表单靠右显示;且由于 h1 | ||
181 | 与表单大小不一,表单太靠上不好看,于是设置二者对齐方式为基线对齐,并取消了 h1 与表单的内外留白。</p> | ||
182 | <p>涉及的 html 模板修改情况如下:</p> | ||
183 | <pre><code>title = f'Directory listing for {displaypath}' | ||
184 | r.append('<!DOCTYPE HTML>') | ||
185 | r.append('<html lang="en">') | ||
186 | r.append('<head>') | ||
187 | r.append(f'<meta charset="{enc}">') | ||
188 | r.append(f'<title>{title}</title>') | ||
189 | r.append('<style>input[type="file"],input[type="submit"] {font-size:1em;position:relative;top:100%;}table{border-collapse:separate;}td,th{padding:0 13px;}.size{text-align:right;}body{font-family:"Courier New",monospace;}form{padding:0;margin:0;}h1{font-size:30px;margin-bottom:0;}a{text-decoration:none;}a:link{color:#0000EE;}a:visited{color:#0000EE;}a:hover{color:#0000EE;}a:active{color:#0000EE;}</style>') | ||
190 | r.append('</head><body>') | ||
191 | r.append('<div style="display:flex;justify-content:space-between;align-items:baseline;">') | ||
192 | r.append(f'<h1>{title}</h1>') | ||
193 | r.append('<form action="" method="post" enctype="multipart/form-data">') | ||
194 | r.append('<input type="file" name="file" id="file" multiple>') | ||
195 | r.append('<input type="submit" value="Upload">') | ||
196 | r.append('</form></div>') | ||
197 | r.append('<hr>')</code></pre> | ||
198 | <p>修改的内容主要是 css 样式和添加了上传表单。</p> | ||
199 | <p>接下来,我们需要在<code>do_POST</code>函数中处理上传的文件。这里我使用了<code>cgi</code>模块,该模块可以帮助我们处理上传的文件。在<code>do_POST</code>函数中,我们需要添加以下代码: | ||
200 | </p> | ||
201 | <pre><code># 注意是class SimpleHTTPRequestHandler的do_POST函数 | ||
202 | # 不要搞错了 | ||
203 | def do_POST(self): | ||
204 | # 获取请求路径 | ||
205 | path=self.translate_path(self.path) | ||
206 | form = cgi.FieldStorage( | ||
207 | fp=self.rfile, | ||
208 | headers=self.headers, | ||
209 | environ={'REQUEST_METHOD': 'POST', | ||
210 | 'CONTENT_TYPE': self.headers['Content-Type'], | ||
211 | }) | ||
212 | |||
213 | files=form['file'] | ||
214 | uploaded_filenames = [] | ||
215 | # 只上传了一个文件和上传了多个文件,是需要分别处理的 | ||
216 | if isinstance(files, list): | ||
217 | for uploaded_file in form['file']: | ||
218 | filename = os.path.join(os.getcwd(), path,uploaded_file.filename) | ||
219 | with open(filename, 'wb') as f: | ||
220 | f.write(uploaded_file.file.read()) | ||
221 | uploaded_filenames.append(uploaded_file.filename) | ||
222 | else: | ||
223 | filename = os.path.join(os.getcwd(), path,files.filename) | ||
224 | with open(filename, 'wb') as f: | ||
225 | f.write(files.file.read()) | ||
226 | uploaded_filenames.append(files.filename) | ||
227 | |||
228 | self.send_response(200) | ||
229 | self.send_header('Content-type', 'text/html') | ||
230 | self.end_headers() | ||
231 | # Format the string with the filenames | ||
232 | html_string = ''' | ||
233 | <!DOCTYPE html> | ||
234 | <html> | ||
235 | <head> | ||
236 | <title>File Upload</title> | ||
237 | <script> | ||
238 | window.onload = function() { | ||
239 | // Display a popup notification | ||
240 | alert("Files uploaded successfully: %s"); | ||
241 | |||
242 | // Refresh the current page with a GET request | ||
243 | window.location.reload(true); | ||
244 | }; | ||
245 | </script> | ||
246 | </head> | ||
247 | <body> | ||
248 | </body> | ||
249 | </html> | ||
250 | ''' % '\n'.join(uploaded_filenames) | ||
251 | |||
252 | # Convert the string to bytes and write it to the response | ||
253 | enc='utf-8' | ||
254 | self.wfile.write(html_string.encode(enc, 'surrogateescape'))</code></pre> | ||
255 | <h1 id="后续">后续</h1> | ||
256 | <p>到了这一步,可以说基础功能完全符合我们的需求了。后续有时间有想法的话,可以加上这么一些功能:</p> | ||
257 | <ul> | ||
258 | <li>自动查看文件类型,将如果属于特定类型的代码,就不采用源文件传回而是使用 html 传回,并使用本网站已有的高亮与一键复制办法,为代码文件进行高亮</li> | ||
259 | <li>对于 html 文件,应当提供查看源码和查看显示效果两种展示方式,供用户选择</li> | ||
260 | <li>能不能在前端就实现文件修改呢……</li> | ||
261 | </ul> | ||
262 | <h1 id="结束">结束</h1> | ||
263 | <p>在经历以上的修改之后,原来的<code>http.server</code>改造成为了一个更加适合免登录浏览服务器文件的工具。再次说明,我修改后的文件<a | ||
264 | href="https://file.qin-juan-ge-zhu.top/useful/myserver.py">在这里</a>。如果你想要使用,可以将其放在<code>/usr/lib/python3.x/http/myserver.py</code>,然后使用以下命令启动: | ||
265 | </p> | ||
266 | <pre><code>python -m http.myserver > py_menu.log 2>&1 &</code></pre> | ||
267 | <p>这条命令会让这个进程在后台运行,并且将运行时的日志和报错信息输出到日志文件中,以备查看和随时确认是否有人在通过这个服务攻击你的服务器(服务连续运行三天,查看日志,你一定会被如此多数量的陌生 ip 访问吓到的)。 | ||
268 | </p> | ||
269 | <p>这个服务默认运行在 8000 端口,你可以按照文件最后的<code>main</code>函数里的参数来修改运行端口。到了这一步,你也可以像为其他网络服务配置 https 一样,使用你的网络服务软件(如 | ||
270 | nginx、apache 之类)为之配置域名和 https 服务了,多好~</p> | ||
271 | <p class="time">2023.1.3晚</p> | ||
272 | <script src="https://www.qin-juan-ge-zhu.top/common/js/comment.js"></script> | ||
273 | </div> | ||
274 | </div> | ||
275 | </body> | ||
276 | |||
277 | </html> \ No newline at end of file | ||
diff --git a/code/linux/httpmyserver.md b/code/linux/httpmyserver.md new file mode 100644 index 0000000..2f9f621 --- /dev/null +++ b/code/linux/httpmyserver.md | |||
@@ -0,0 +1,291 @@ | |||
1 | <!-- python实现服务器文件浏览 --> | ||
2 | |||
3 | 有了服务器,我们自然会有这样的需求:**把一些内容传到服务器上,方便其他人访问或自己在其他设备(如手机)上,通过浏览器直接访问**。我们需要它能完成以下功能: | ||
4 | |||
5 | - 能让我们自己点进(或退出)层层文件夹,展示当前文件夹下的所有文件/文件夹 | ||
6 | - 点击文件,能够看到文件内容 | ||
7 | |||
8 |  | ||
9 | |||
10 | 点击进入文件`nginx.txt`,我们要求看到文件的内容: | ||
11 | |||
12 |  | ||
13 | |||
14 | 既要看文件内容,又要能浏览不同的文件夹。这个功能如果直接用程序或 nginx 实现似乎有些新手不友好。没事,python3 为我们提供了这个功能,脚本为`/usr/lib/python3.x/http/server.py`。使用时执行以下命令即可: | ||
15 | |||
16 | ```bash | ||
17 | python3 -m http.server | ||
18 | ``` | ||
19 | |||
20 | 但 python 提供的脚本有些不尽人意: | ||
21 | |||
22 | - 遇到当前路径有`index.html`就展示网页,而不是展示当前路径下的文件 | ||
23 | - 无法回退到上层文件夹 | ||
24 | - 无法查看文件详细信息 | ||
25 | - 浏览普通文件时,响应头没有指定文件编码,导致文件显示乱码 | ||
26 | - 无法方便地上传文件 | ||
27 | |||
28 | 处于以上种种原因,我对该程序做了一点点修改。修改后的脚本我选择放在`/usr/lib/python3.x/http/myserver.py`(和原有`server.py`放在一起)。代码全文我放在了[这里](https://file.qin-juan-ge-zhu.top/useful/myserver.py)。在这里仅对修改的部分加以说明。 | ||
29 | |||
30 | 首先,为了完成以下功能,需要引入几个新的模块: | ||
31 | |||
32 | ```python | ||
33 | import cgi | ||
34 | import pwd | ||
35 | import grp | ||
36 | import time | ||
37 | import stat | ||
38 | import math | ||
39 | ``` | ||
40 | |||
41 | # 遇到 index.html 时 | ||
42 | |||
43 | 在`http.server`(源码路径`/usr/lib/python3.x/http/server.py`)第 710 行(`def send_head`函数)中有这样一段: | ||
44 | |||
45 | ```python | ||
46 | for index in "index.html", "index.htm": | ||
47 | index = os.path.join(path, index) | ||
48 | if os.path.isfile(index): | ||
49 | path = index | ||
50 | break | ||
51 | ``` | ||
52 | |||
53 | 可以看到,该段代码的作用就是当请求的路径是一个文件夹且该文件夹下有`index.html`或`index.htm`,则将该文件作为响应内容返回。这一点对于一般的网页服务器来说是不错的,但咱们需要的是浏览服务器文件,而不是展示网页根目录,所以不需要这个功能,该段代码删除即可。 | ||
54 | |||
55 | # 浏览文件出现乱码 | ||
56 | |||
57 | 你可能会问,上边看到的 html 模板不是已经明文规定编码是`utf-8`了吗,,怎么还会乱码?没错,html 是说明编码了,但我们除了浏览以 html 形式展示的目录连接之外,还是要看文件内容的,文件里或多或少都会有点中文不是吗?对于文件而言,编码必须由 http 的响应头指出,否则客户端就会根据文件内容猜测其编码然后展示出来,就成了乱码。 | ||
58 | |||
59 | ```python | ||
60 | # http.server, line 763 | ||
61 | # also in the function 'send_head' | ||
62 | # this line and its context are about http response header | ||
63 | self.send_header("Content-type", ctype) | ||
64 | # change it into the below | ||
65 | self.send_header("Content-type", ctype + "; charset=utf-8") | ||
66 | ``` | ||
67 | |||
68 | # 展示所有文件和文件夹 | ||
69 | |||
70 | 在`server.py`第 772 行,有一个`list_directory`函数,是用来展示当前路径下有哪些文件或文件夹的,但是这有个缺陷,就是只能点进下一级文件夹,但无法返回上一级。为此,我们需要做一些修改: | ||
71 | |||
72 | ```python | ||
73 | def list_directory(self, path): | ||
74 | try: | ||
75 | list = os.listdir(path) | ||
76 | except OSError: | ||
77 | self.send_error( | ||
78 | HTTPStatus.NOT_FOUND, | ||
79 | "No permission to list directory") | ||
80 | return None | ||
81 | # 添加这一行,使得可以返回上一级和维持本级 | ||
82 | # 以使展示更贴近于Linux的展示 | ||
83 | list.extend([".",".."]) | ||
84 | # 排序方式原来是完全按照字母顺序,不甚方便 | ||
85 | # 修改为文件夹排在文件之前,各自按照字母排序 | ||
86 | # list.sort(key=lambda a: .lower()) | ||
87 | list.sort(key=lambda a: (not os.path.isdir(os.path.join(path,a)),a.lower())) | ||
88 | ``` | ||
89 | |||
90 | 可能你已经注意到了,主要的展示目录的界面就是这个函数底下的 html 模板。下面我们会经常修改这个模板的。 | ||
91 | |||
92 | # 文件详细信息展示 | ||
93 | |||
94 | 有时候我们需要的不仅仅是文件本身,还需要关注文件的详细信息,如用户权限、文件大小、修改日期之类的。这些信息在 Linux 系统中通常使用命令`ls -l`或其别名`ll`来查看。我想要模仿这个命令的输出格式,做一些微调: | ||
95 | |||
96 | - 第一列是文件名,是一个超链接,这个不必说 | ||
97 | - 第二列是目录类型和其权限位 | ||
98 | - 第三列是目录的硬链接数目,文件是 1,目录则是其下的文件和文件夹总数(包括`.`和`..`) | ||
99 | - 第四列、第五列分别是文件的所有者和所属组 | ||
100 | - 第六列文件大小,按照人可读的方式展示(也就是使用适当的单位) | ||
101 | - 第七、第八列是创建时间和修改时间 | ||
102 | |||
103 | 为了实现以上目的,我们不能继续使用无序列表来展示信息,而是应当改为表格;同时需要注意的是: | ||
104 | |||
105 | - 既然要展示权限位,必然需要使用等宽字体。这里我选择整个 html 均使用等宽字体 | ||
106 | - 为了美观,目录超链接的样式我做了一些修改,主要是去除了其下划线、设置无论是否访问过都显示为同一个颜色 | ||
107 | - 还做了其他一些小的样式修改 | ||
108 | |||
109 | ```python | ||
110 | enc = sys.getfilesystemencoding() | ||
111 | title = f'Directory listing for {displaypath}' | ||
112 | r.append('<!DOCTYPE HTML>') | ||
113 | r.append('<html lang="en">') | ||
114 | r.append('<head>') | ||
115 | r.append(f'<meta charset="{enc}">') | ||
116 | r.append(f'<title>{title}</title>') | ||
117 | r.append('table{border-collapse:separate;}td,th{padding:0 13px;}.size{text-align:right;}body{font-family:"Courier New",monospace;}h1{font-size:30px;margin-bottom:0;}a{text-decoration:none;}a:link{color:#0000EE;}a:visited{color:#0000EE;}a:hover{color:#0000EE;}a:active{color:#0000EE;}</style>') | ||
118 | r.append('</head><body>') | ||
119 | r.append(f'<h1>{title}</h1>') | ||
120 | r.append('<hr>') | ||
121 | r.append('<table><tr><th>Name</th><th>Permissions</th><th>Hard Links</th><th>Owner</th><th>Group</th><th>Size</th><th>Creation Time</th><th>Last Modified</th></tr>') | ||
122 | for name in list: | ||
123 | fullname = os.path.join(path, name) | ||
124 | displayname = linkname = name | ||
125 | # Append / for directories or @ for symbolic links | ||
126 | if os.path.isdir(fullname): | ||
127 | displayname = name + "/" | ||
128 | linkname = name + "/" | ||
129 | if os.path.islink(fullname): | ||
130 | displayname = name + "@" | ||
131 | # Note: a link to a directory displays with @ and links with / | ||
132 | |||
133 | # Add detailed information for each file/directory | ||
134 | stat_info = os.stat(fullname) | ||
135 | mode = stat.filemode(stat_info.st_mode) | ||
136 | hard_links = stat_info.st_nlink | ||
137 | owner = pwd.getpwuid(stat_info.st_uid).pw_name | ||
138 | group = grp.getgrgid(stat_info.st_gid).gr_name | ||
139 | size_bytes = stat_info.st_size | ||
140 | # 文件大小需要可读 | ||
141 | if size_bytes == 0: | ||
142 | size = '0B' | ||
143 | size_name = (' B', 'KB', 'MB', 'GB') | ||
144 | i = int(math.floor(math.log(size_bytes, 1024))) | ||
145 | p = math.pow(1024, i) | ||
146 | s = size_bytes / p | ||
147 | size = '%.2f %s' % (s, size_name[i]) | ||
148 | creation_time = time.strftime('%Y-%m-%d %H:%M:%S', time.localtime(stat_info.st_ctime)) | ||
149 | last_modified = time.strftime('%Y-%m-%d %H:%M:%S', time.localtime(stat_info.st_mtime)) | ||
150 | |||
151 | r.append('<tr><td><a href="%s">%s</a></td><td>%s</td><td>%d</td><td>%s</td><td>%s</td><td class="size">%s</td><td>%s</td><td>%s</td></tr>' | ||
152 | % (urllib.parse.quote(linkname, errors='surrogatepass'), | ||
153 | html.escape(displayname, quote=False), | ||
154 | mode, hard_links, owner, group, size, creation_time, last_modified)) | ||
155 | r.append('</table>\n</body>\n') | ||
156 | r.append(''' | ||
157 | <script> | ||
158 | onload = () => { | ||
159 | document.getElementById('file').addEventListener('change', function() { | ||
160 | var files = this.files; | ||
161 | var fileNames = ''; | ||
162 | for (var i = 0; i < files.length; i++) { | ||
163 | fileNames += files[i].name + '\\n'; | ||
164 | } | ||
165 | confirm('Are you sure you want to upload these files?\\n' + fileNames); | ||
166 | }); | ||
167 | } | ||
168 | </script> | ||
169 | ''') | ||
170 | r.append('</html>\n') | ||
171 | encoded = '\n'.join(r).encode(enc, 'surrogateescape') | ||
172 | f = io.BytesIO() | ||
173 | f.write(encoded) | ||
174 | f.seek(0) | ||
175 | self.send_response(HTTPStatus.OK) | ||
176 | self.send_header("Content-type", "text/html; charset=%s" % enc) | ||
177 | self.send_header("Content-Length", str(len(encoded))) | ||
178 | self.end_headers() | ||
179 | return f | ||
180 | ``` | ||
181 | |||
182 | 到了这一步,整体的布局美观了不少,可以称得上是赏心悦目了。 | ||
183 | |||
184 | # 文件上传 | ||
185 | |||
186 | 其实轮起来,文件上传这个功能我本是不需要的,我所有设备都能通过`scp`将文件上传到服务器上。这一切,都是被微信逼的。 | ||
187 | |||
188 | 前两天,东方直心前辈想要用微信给我发他最新修改的《毛泽东大传》,一连几次打包发我,我都没收到,只能解释为被微信截胡了。没办法,我又不能教老人用什么太繁杂的方法,因而一连几天搞这个文件上传功能,最终在黄四郎同志和 gpt、copilot 的帮助下,完成了这个功能。 | ||
189 | |||
190 | 首先,我们需要一个按钮来供使用者上传文件。为了整体页面协调,我选择将上传表单与 h1 放在同一行,都在分割线上方,表单靠右显示;且由于 h1 与表单大小不一,表单太靠上不好看,于是设置二者对齐方式为基线对齐,并取消了 h1 与表单的内外留白。 | ||
191 | |||
192 | 涉及的 html 模板修改情况如下: | ||
193 | |||
194 | ```python | ||
195 | title = f'Directory listing for {displaypath}' | ||
196 | r.append('<!DOCTYPE HTML>') | ||
197 | r.append('<html lang="en">') | ||
198 | r.append('<head>') | ||
199 | r.append(f'<meta charset="{enc}">') | ||
200 | r.append(f'<title>{title}</title>') | ||
201 | r.append('<style>input[type="file"],input[type="submit"] {font-size:1em;position:relative;top:100%;}table{border-collapse:separate;}td,th{padding:0 13px;}.size{text-align:right;}body{font-family:"Courier New",monospace;}form{padding:0;margin:0;}h1{font-size:30px;margin-bottom:0;}a{text-decoration:none;}a:link{color:#0000EE;}a:visited{color:#0000EE;}a:hover{color:#0000EE;}a:active{color:#0000EE;}</style>') | ||
202 | r.append('</head><body>') | ||
203 | r.append('<div style="display:flex;justify-content:space-between;align-items:baseline;">') | ||
204 | r.append(f'<h1>{title}</h1>') | ||
205 | r.append('<form action="" method="post" enctype="multipart/form-data">') | ||
206 | r.append('<input type="file" name="file" id="file" multiple>') | ||
207 | r.append('<input type="submit" value="Upload">') | ||
208 | r.append('</form></div>') | ||
209 | r.append('<hr>') | ||
210 | ``` | ||
211 | |||
212 | 修改的内容主要是 css 样式和添加了上传表单。 | ||
213 | |||
214 | 接下来,我们需要在`do_POST`函数中处理上传的文件。这里我使用了`cgi`模块,该模块可以帮助我们处理上传的文件。在`do_POST`函数中,我们需要添加以下代码: | ||
215 | |||
216 | ```python | ||
217 | # 注意是class SimpleHTTPRequestHandler的do_POST函数 | ||
218 | # 不要搞错了 | ||
219 | def do_POST(self): | ||
220 | # 获取请求路径 | ||
221 | path=self.translate_path(self.path) | ||
222 | form = cgi.FieldStorage( | ||
223 | fp=self.rfile, | ||
224 | headers=self.headers, | ||
225 | environ={'REQUEST_METHOD': 'POST', | ||
226 | 'CONTENT_TYPE': self.headers['Content-Type'], | ||
227 | }) | ||
228 | |||
229 | files=form['file'] | ||
230 | uploaded_filenames = [] | ||
231 | # 只上传了一个文件和上传了多个文件,是需要分别处理的 | ||
232 | if isinstance(files, list): | ||
233 | for uploaded_file in form['file']: | ||
234 | filename = os.path.join(os.getcwd(), path,uploaded_file.filename) | ||
235 | with open(filename, 'wb') as f: | ||
236 | f.write(uploaded_file.file.read()) | ||
237 | uploaded_filenames.append(uploaded_file.filename) | ||
238 | else: | ||
239 | filename = os.path.join(os.getcwd(), path,files.filename) | ||
240 | with open(filename, 'wb') as f: | ||
241 | f.write(files.file.read()) | ||
242 | uploaded_filenames.append(files.filename) | ||
243 | |||
244 | self.send_response(200) | ||
245 | self.send_header('Content-type', 'text/html') | ||
246 | self.end_headers() | ||
247 | # Format the string with the filenames | ||
248 | html_string = ''' | ||
249 | <!DOCTYPE html> | ||
250 | <html> | ||
251 | <head> | ||
252 | <title>File Upload</title> | ||
253 | <script> | ||
254 | window.onload = function() { | ||
255 | // Display a popup notification | ||
256 | alert("Files uploaded successfully: %s"); | ||
257 | |||
258 | // Refresh the current page with a GET request | ||
259 | window.location.reload(true); | ||
260 | }; | ||
261 | </script> | ||
262 | </head> | ||
263 | <body> | ||
264 | </body> | ||
265 | </html> | ||
266 | ''' % '\n'.join(uploaded_filenames) | ||
267 | |||
268 | # Convert the string to bytes and write it to the response | ||
269 | enc='utf-8' | ||
270 | self.wfile.write(html_string.encode(enc, 'surrogateescape')) | ||
271 | ``` | ||
272 | |||
273 | # 后续 | ||
274 | |||
275 | 到了这一步,可以说基础功能完全符合我们的需求了。后续有时间有想法的话,可以加上这么一些功能: | ||
276 | |||
277 | - 自动查看文件类型,将如果属于特定类型的代码,就不采用源文件传回而是使用 html 传回,并使用本网站已有的高亮与一键复制办法,为代码文件进行高亮 | ||
278 | - 对于 html 文件,应当提供查看源码和查看显示效果两种展示方式,供用户选择 | ||
279 | - 能不能在前端就实现文件修改呢…… | ||
280 | |||
281 | # 结束 | ||
282 | |||
283 | 在经历以上的修改之后,原来的`http.server`改造成为了一个更加适合免登录浏览服务器文件的工具。再次说明,我修改后的文件[在这里](https://file.qin-juan-ge-zhu.top/useful/myserver.py)。如果你想要使用,可以将其放在`/usr/lib/python3.x/http/myserver.py`,然后使用以下命令启动: | ||
284 | |||
285 | ```bash | ||
286 | python -m http.myserver > py_menu.log 2>&1 & | ||
287 | ``` | ||
288 | |||
289 | 这条命令会让这个进程在后台运行,并且将运行时的日志和报错信息输出到日志文件中,以备查看和随时确认是否有人在通过这个服务攻击你的服务器(服务连续运行三天,查看日志,你一定会被如此多数量的陌生 ip 访问吓到的)。 | ||
290 | |||
291 | 这个服务默认运行在 8000 端口,你可以按照文件最后的`main`函数里的参数来修改运行端口。到了这一步,你也可以像为其他网络服务配置 https 一样,使用你的网络服务软件(如 nginx、apache 之类)为之配置域名和 https 服务了,多好~ | ||
diff --git a/code/linux/vim.html b/code/linux/vim.html index 22f070f..fc1f145 100644 --- a/code/linux/vim.html +++ b/code/linux/vim.html | |||
@@ -3,8 +3,9 @@ | |||
3 | 3 | ||
4 | <head> | 4 | <head> |
5 | <meta charset="utf-8" /> | 5 | <meta charset="utf-8" /> |
6 | <meta name="generator" content="pandoc" /> | ||
6 | <meta name="viewport" content="width=device-width, initial-scale=1.0, user-scalable=yes" /> | 7 | <meta name="viewport" content="width=device-width, initial-scale=1.0, user-scalable=yes" /> |
7 | <title>vim学习与配置</title> | 8 | <title>vim学习与我的配置</title> |
8 | <link rel="stylesheet" href="https://www.qin-juan-ge-zhu.top/common/CSS/pandoc.css"> | 9 | <link rel="stylesheet" href="https://www.qin-juan-ge-zhu.top/common/CSS/pandoc.css"> |
9 | <script type="text/javascript" src="https://www.qin-juan-ge-zhu.top/common/script4code.js"></script> | 10 | <script type="text/javascript" src="https://www.qin-juan-ge-zhu.top/common/script4code.js"></script> |
10 | </head> | 11 | </head> |
@@ -18,388 +19,45 @@ | |||
18 | 时代码上下反复跳转,令你烦闷否?别人写的编辑器设置种种不便,是否想过有一个自己私人订制的编辑器?如果你有以上苦恼,那么 vim 将是你的绝佳选择。</p> | 19 | 时代码上下反复跳转,令你烦闷否?别人写的编辑器设置种种不便,是否想过有一个自己私人订制的编辑器?如果你有以上苦恼,那么 vim 将是你的绝佳选择。</p> |
19 | <p>vim 是一个文本编辑器,素有“编辑器之神”的美名,同时它也是 Linux 尤其是服务器环境下编辑文件的主要方法。</p> | 20 | <p>vim 是一个文本编辑器,素有“编辑器之神”的美名,同时它也是 Linux 尤其是服务器环境下编辑文件的主要方法。</p> |
20 | <p>vim 的主要特点是分模式编辑:</p> | 21 | <p>vim 的主要特点是分模式编辑:</p> |
21 | <div class="table-div"> | 22 | <table> |
22 | <table> | 23 | <thead> |
23 | <thead> | 24 | <tr class="header"> |
24 | <tr class="header"> | 25 | <th>模式</th> |
25 | <th>模式</th> | 26 | <th>说明</th> |
26 | <th>说明</th> | 27 | <th>功能</th> |
27 | <th>功能</th> | 28 | </tr> |
28 | </tr> | 29 | </thead> |
29 | </thead> | 30 | <tbody> |
30 | <tbody> | 31 | <tr class="odd"> |
31 | <tr class="odd"> | 32 | <td>Normal</td> |
32 | <td>Normal</td> | 33 | <td>正常</td> |
33 | <td>正常</td> | 34 | <td>光标移动、阅读文件</td> |
34 | <td>光标移动、阅读文件</td> | 35 | </tr> |
35 | </tr> | 36 | <tr class="even"> |
36 | <tr class="even"> | 37 | <td>Insert</td> |
37 | <td>Insert</td> | 38 | <td>插入</td> |
38 | <td>插入</td> | 39 | <td>在当前光标位置插入内容,编辑文件</td> |
39 | <td>在当前光标位置插入内容,编辑文件</td> | 40 | </tr> |
40 | </tr> | 41 | <tr class="odd"> |
41 | <tr class="odd"> | 42 | <td>Visual</td> |
42 | <td>Visual</td> | 43 | <td>视觉/选择</td> |
43 | <td>视觉/选择</td> | 44 | <td>选中、复制粘贴</td> |
44 | <td>选中、复制粘贴</td> | 45 | </tr> |
45 | </tr> | 46 | <tr class="even"> |
46 | <tr class="even"> | 47 | <td>Replace</td> |
47 | <td>Replace</td> | 48 | <td>替换</td> |
48 | <td>替换</td> | 49 | <td>替换当前光标下字符</td> |
49 | <td>替换当前光标下字符</td> | 50 | </tr> |
50 | </tr> | 51 | </tbody> |
51 | </tbody> | 52 | </table> |
52 | </table> | ||
53 | </div> | ||
54 | <p>其关系如下:</p> | 53 | <p>其关系如下:</p> |
55 | <!-- 楚河汉界 --> | 54 | <pre><code>graph LR |
56 | <div class="md-diagram-panel md-fences-adv-panel" style="text-align: center;"><svg id="mermaidChart0" | 55 | N(Normal)--->I(Insert) |
57 | width="100%" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" | 56 | N--->V(Visual) |
58 | style="max-width: 262.82501220703125px;display: inline;" | 57 | N--->R(Replace) |
59 | viewBox="-8 -7.999998092651367 262.82501220703125 237.79998779296875" | 58 | I-->N |
60 | role="graphics-document document" aria-roledescription="flowchart-v2"> | 59 | V-->N |
61 | <style> | 60 | R-->N</code></pre> |
62 | #mermaidChart0 { | ||
63 | font-family: sans-serif; | ||
64 | font-size: 16px; | ||
65 | fill: #333; | ||
66 | } | ||
67 | |||
68 | #mermaidChart0 .error-icon { | ||
69 | fill: #552222; | ||
70 | } | ||
71 | |||
72 | #mermaidChart0 .error-text { | ||
73 | fill: #552222; | ||
74 | stroke: #552222; | ||
75 | } | ||
76 | |||
77 | #mermaidChart0 .edge-thickness-normal { | ||
78 | stroke-width: 2px; | ||
79 | } | ||
80 | |||
81 | #mermaidChart0 .edge-thickness-thick { | ||
82 | stroke-width: 3.5px; | ||
83 | } | ||
84 | |||
85 | #mermaidChart0 .edge-pattern-solid { | ||
86 | stroke-dasharray: 0; | ||
87 | } | ||
88 | |||
89 | #mermaidChart0 .edge-pattern-dashed { | ||
90 | stroke-dasharray: 3; | ||
91 | } | ||
92 | |||
93 | #mermaidChart0 .edge-pattern-dotted { | ||
94 | stroke-dasharray: 2; | ||
95 | } | ||
96 | |||
97 | #mermaidChart0 .marker { | ||
98 | fill: #333333; | ||
99 | stroke: #333333; | ||
100 | } | ||
101 | |||
102 | #mermaidChart0 .marker.cross { | ||
103 | stroke: #333333; | ||
104 | } | ||
105 | |||
106 | #mermaidChart0 svg { | ||
107 | font-family: sans-serif; | ||
108 | font-size: 16px; | ||
109 | } | ||
110 | |||
111 | #mermaidChart0 .label { | ||
112 | font-family: sans-serif; | ||
113 | color: #333; | ||
114 | } | ||
115 | |||
116 | #mermaidChart0 .cluster-label text { | ||
117 | fill: #333; | ||
118 | } | ||
119 | |||
120 | #mermaidChart0 .cluster-label span, | ||
121 | #mermaidChart0 p { | ||
122 | color: #333; | ||
123 | } | ||
124 | |||
125 | #mermaidChart0 .label text, | ||
126 | #mermaidChart0 span, | ||
127 | #mermaidChart0 p { | ||
128 | fill: #333; | ||
129 | color: #333; | ||
130 | } | ||
131 | |||
132 | #mermaidChart0 .node rect, | ||
133 | #mermaidChart0 .node circle, | ||
134 | #mermaidChart0 .node ellipse, | ||
135 | #mermaidChart0 .node polygon, | ||
136 | #mermaidChart0 .node path { | ||
137 | fill: #ECECFF; | ||
138 | stroke: #9370DB; | ||
139 | stroke-width: 1px; | ||
140 | } | ||
141 | |||
142 | #mermaidChart0 .flowchart-label text { | ||
143 | text-anchor: middle; | ||
144 | } | ||
145 | |||
146 | #mermaidChart0 .node .label { | ||
147 | text-align: center; | ||
148 | } | ||
149 | |||
150 | #mermaidChart0 .node.clickable { | ||
151 | cursor: pointer; | ||
152 | } | ||
153 | |||
154 | #mermaidChart0 .arrowheadPath { | ||
155 | fill: #333333; | ||
156 | } | ||
157 | |||
158 | #mermaidChart0 .edgePath .path { | ||
159 | stroke: #333333; | ||
160 | stroke-width: 2.0px; | ||
161 | } | ||
162 | |||
163 | #mermaidChart0 .flowchart-link { | ||
164 | stroke: #333333; | ||
165 | fill: none; | ||
166 | } | ||
167 | |||
168 | #mermaidChart0 .edgeLabel { | ||
169 | background-color: #e8e8e8; | ||
170 | text-align: center; | ||
171 | } | ||
172 | |||
173 | #mermaidChart0 .edgeLabel rect { | ||
174 | opacity: 0.5; | ||
175 | background-color: #e8e8e8; | ||
176 | fill: #e8e8e8; | ||
177 | } | ||
178 | |||
179 | #mermaidChart0 .cluster rect { | ||
180 | fill: #ffffde; | ||
181 | stroke: #aaaa33; | ||
182 | stroke-width: 1px; | ||
183 | } | ||
184 | |||
185 | #mermaidChart0 .cluster text { | ||
186 | fill: #333; | ||
187 | } | ||
188 | |||
189 | #mermaidChart0 .cluster span, | ||
190 | #mermaidChart0 p { | ||
191 | color: #333; | ||
192 | } | ||
193 | |||
194 | #mermaidChart0 div.mermaidTooltip { | ||
195 | position: absolute; | ||
196 | text-align: center; | ||
197 | max-width: 200px; | ||
198 | padding: 2px; | ||
199 | font-family: sans-serif; | ||
200 | font-size: 12px; | ||
201 | background: hsl(80, 100%, 96.2745098039%); | ||
202 | border: 1px solid #aaaa33; | ||
203 | border-radius: 2px; | ||
204 | pointer-events: none; | ||
205 | z-index: 100; | ||
206 | } | ||
207 | |||
208 | #mermaidChart0 .flowchartTitleText { | ||
209 | text-anchor: middle; | ||
210 | font-size: 18px; | ||
211 | fill: #333; | ||
212 | } | ||
213 | |||
214 | #mermaidChart0 :root { | ||
215 | --mermaid-alt-font-family: sans-serif; | ||
216 | } | ||
217 | </style> | ||
218 | <g> | ||
219 | <marker id="flowchart-pointEnd" class="marker flowchart" viewBox="0 0 12 20" refX="10" refY="5" | ||
220 | markerUnits="userSpaceOnUse" markerWidth="12" markerHeight="12" orient="auto"> | ||
221 | <path d="M 0 0 L 10 5 L 0 10 z" class="arrowMarkerPath" | ||
222 | style="stroke-width: 1; stroke-dasharray: 1, 0;"></path> | ||
223 | </marker> | ||
224 | <marker id="flowchart-pointStart" class="marker flowchart" viewBox="0 0 10 10" refX="0" refY="5" | ||
225 | markerUnits="userSpaceOnUse" markerWidth="12" markerHeight="12" orient="auto"> | ||
226 | <path d="M 0 5 L 10 10 L 10 0 z" class="arrowMarkerPath" | ||
227 | style="stroke-width: 1; stroke-dasharray: 1, 0;"></path> | ||
228 | </marker> | ||
229 | <marker id="flowchart-circleEnd" class="marker flowchart" viewBox="0 0 10 10" refX="11" refY="5" | ||
230 | markerUnits="userSpaceOnUse" markerWidth="11" markerHeight="11" orient="auto"> | ||
231 | <circle cx="5" cy="5" r="5" class="arrowMarkerPath" | ||
232 | style="stroke-width: 1; stroke-dasharray: 1, 0;"></circle> | ||
233 | </marker> | ||
234 | <marker id="flowchart-circleStart" class="marker flowchart" viewBox="0 0 10 10" refX="-1" | ||
235 | refY="5" markerUnits="userSpaceOnUse" markerWidth="11" markerHeight="11" orient="auto"> | ||
236 | <circle cx="5" cy="5" r="5" class="arrowMarkerPath" | ||
237 | style="stroke-width: 1; stroke-dasharray: 1, 0;"></circle> | ||
238 | </marker> | ||
239 | <marker id="flowchart-crossEnd" class="marker cross flowchart" viewBox="0 0 11 11" refX="12" | ||
240 | refY="5.2" markerUnits="userSpaceOnUse" markerWidth="11" markerHeight="11" orient="auto"> | ||
241 | <path d="M 1,1 l 9,9 M 10,1 l -9,9" class="arrowMarkerPath" | ||
242 | style="stroke-width: 2; stroke-dasharray: 1, 0;"></path> | ||
243 | </marker> | ||
244 | <marker id="flowchart-crossStart" class="marker cross flowchart" viewBox="0 0 11 11" refX="-1" | ||
245 | refY="5.2" markerUnits="userSpaceOnUse" markerWidth="11" markerHeight="11" orient="auto"> | ||
246 | <path d="M 1,1 l 9,9 M 10,1 l -9,9" class="arrowMarkerPath" | ||
247 | style="stroke-width: 2; stroke-dasharray: 1, 0;"></path> | ||
248 | </marker> | ||
249 | <g class="root"> | ||
250 | <g class="clusters"></g> | ||
251 | <g class="edgePaths"> | ||
252 | <path | ||
253 | d="M48.526970990449726,90.5999984741211L97.36250305175781,10.299999237060547L122.36250305175781,10.299999237060547L147.3625030517578,10.299999237060547L180.28125381469727,15.589745202208903" | ||
254 | id="L-N-I-0" | ||
255 | class=" edge-thickness-normal edge-pattern-solid flowchart-link LS-N LE-I" | ||
256 | style="fill:none;" marker-end="url(#flowchart-pointEnd)"></path> | ||
257 | <path | ||
258 | d="M71.36480354119537,90.5999984741211L97.36250305175781,75.5999984741211L122.36250305175781,75.5999984741211L147.3625030517578,75.5999984741211L179.36875343322754,93.75519542413791" | ||
259 | id="L-N-V-0" | ||
260 | class=" edge-thickness-normal edge-pattern-solid flowchart-link LS-N LE-V" | ||
261 | style="fill:none;" marker-end="url(#flowchart-pointEnd)"></path> | ||
262 | <path | ||
263 | d="M58.64019142520192,131.1999969482422L97.36250305175781,166.1999969482422L122.36250305175781,166.1999969482422L147.3625030517578,166.1999969482422L173.806377885162,181.1999969482422" | ||
264 | id="L-N-R-0" | ||
265 | class=" edge-thickness-normal edge-pattern-solid flowchart-link LS-N LE-R" | ||
266 | style="fill:none;" marker-end="url(#flowchart-pointEnd)"></path> | ||
267 | <path | ||
268 | d="M180.28125381469727,36.92719562072298L147.3625030517578,55.599998474121094L122.36250305175781,55.599998474121094L97.36250305175781,55.599998474121094L58.64019142520192,90.5999984741211" | ||
269 | id="L-I-N-0" | ||
270 | class=" edge-thickness-normal edge-pattern-solid flowchart-link LS-I LE-N" | ||
271 | style="fill:none;" marker-end="url(#flowchart-pointEnd)"></path> | ||
272 | <path | ||
273 | d="M179.36875343322754,128.04479999822536L147.3625030517578,146.1999969482422L122.36250305175781,146.1999969482422L97.36250305175781,146.1999969482422L71.36480354119537,131.1999969482422" | ||
274 | id="L-V-N-0" | ||
275 | class=" edge-thickness-normal edge-pattern-solid flowchart-link LS-V LE-N" | ||
276 | style="fill:none;" marker-end="url(#flowchart-pointEnd)"></path> | ||
277 | <path | ||
278 | d="M172.3625030517578,207.48272195515207L147.3625030517578,211.49999618530273L122.36250305175781,211.49999618530273L97.36250305175781,211.49999618530273L48.526970990449726,131.1999969482422" | ||
279 | id="L-R-N-0" | ||
280 | class=" edge-thickness-normal edge-pattern-solid flowchart-link LS-R LE-N" | ||
281 | style="fill:none;" marker-end="url(#flowchart-pointEnd)"></path> | ||
282 | </g> | ||
283 | <g class="edgeLabels"> | ||
284 | <g class="edgeLabel"> | ||
285 | <g class="label" transform="translate(0, 0)"> | ||
286 | <foreignObject width="0" height="0"> | ||
287 | <div xmlns="http://www.w3.org/1999/xhtml" | ||
288 | style="display: inline-block; white-space: nowrap;"><span | ||
289 | class="edgeLabel"></span></div> | ||
290 | </foreignObject> | ||
291 | </g> | ||
292 | </g> | ||
293 | <g class="edgeLabel"> | ||
294 | <g class="label" transform="translate(0, 0)"> | ||
295 | <foreignObject width="0" height="0"> | ||
296 | <div xmlns="http://www.w3.org/1999/xhtml" | ||
297 | style="display: inline-block; white-space: nowrap;"><span | ||
298 | class="edgeLabel"></span></div> | ||
299 | </foreignObject> | ||
300 | </g> | ||
301 | </g> | ||
302 | <g class="edgeLabel"> | ||
303 | <g class="label" transform="translate(0, 0)"> | ||
304 | <foreignObject width="0" height="0"> | ||
305 | <div xmlns="http://www.w3.org/1999/xhtml" | ||
306 | style="display: inline-block; white-space: nowrap;"><span | ||
307 | class="edgeLabel"></span></div> | ||
308 | </foreignObject> | ||
309 | </g> | ||
310 | </g> | ||
311 | <g class="edgeLabel"> | ||
312 | <g class="label" transform="translate(0, 0)"> | ||
313 | <foreignObject width="0" height="0"> | ||
314 | <div xmlns="http://www.w3.org/1999/xhtml" | ||
315 | style="display: inline-block; white-space: nowrap;"><span | ||
316 | class="edgeLabel"></span></div> | ||
317 | </foreignObject> | ||
318 | </g> | ||
319 | </g> | ||
320 | <g class="edgeLabel"> | ||
321 | <g class="label" transform="translate(0, 0)"> | ||
322 | <foreignObject width="0" height="0"> | ||
323 | <div xmlns="http://www.w3.org/1999/xhtml" | ||
324 | style="display: inline-block; white-space: nowrap;"><span | ||
325 | class="edgeLabel"></span></div> | ||
326 | </foreignObject> | ||
327 | </g> | ||
328 | </g> | ||
329 | <g class="edgeLabel"> | ||
330 | <g class="label" transform="translate(0, 0)"> | ||
331 | <foreignObject width="0" height="0"> | ||
332 | <div xmlns="http://www.w3.org/1999/xhtml" | ||
333 | style="display: inline-block; white-space: nowrap;"><span | ||
334 | class="edgeLabel"></span></div> | ||
335 | </foreignObject> | ||
336 | </g> | ||
337 | </g> | ||
338 | </g> | ||
339 | <g class="nodes"> | ||
340 | <g class="node default default flowchart-label" id="flowchart-N-24" | ||
341 | transform="translate(36.181251525878906, 110.89999771118164)"> | ||
342 | <rect class="basic label-container" style="" rx="5" ry="5" x="-36.18124961853027" | ||
343 | y="-20.300000190734863" width="72.36249923706055" height="40.60000038146973"> | ||
344 | </rect> | ||
345 | <g class="label" style="" | ||
346 | transform="translate(-28.681249618530273, -12.800000190734863)"> | ||
347 | <rect></rect> | ||
348 | <foreignObject width="57.36249923706055" height="25.600000381469727"> | ||
349 | <div xmlns="http://www.w3.org/1999/xhtml" | ||
350 | style="display: inline-block; white-space: nowrap;"><span | ||
351 | class="nodeLabel">Normal</span></div> | ||
352 | </foreignObject> | ||
353 | </g> | ||
354 | </g> | ||
355 | <g class="node default default flowchart-label" id="flowchart-I-25" | ||
356 | transform="translate(209.59375381469727, 20.299999237060547)"> | ||
357 | <rect class="basic label-container" style="" rx="5" ry="5" x="-29.3125" | ||
358 | y="-20.300000190734863" width="58.625" height="40.60000038146973"></rect> | ||
359 | <g class="label" style="" transform="translate(-21.8125, -12.800000190734863)"> | ||
360 | <rect></rect> | ||
361 | <foreignObject width="43.625" height="25.600000381469727"> | ||
362 | <div xmlns="http://www.w3.org/1999/xhtml" | ||
363 | style="display: inline-block; white-space: nowrap;"><span | ||
364 | class="nodeLabel">Insert</span></div> | ||
365 | </foreignObject> | ||
366 | </g> | ||
367 | </g> | ||
368 | <g class="node default default flowchart-label" id="flowchart-V-27" | ||
369 | transform="translate(209.59375381469727, 110.89999771118164)"> | ||
370 | <rect class="basic label-container" style="" rx="5" ry="5" x="-30.225000381469727" | ||
371 | y="-20.300000190734863" width="60.45000076293945" height="40.60000038146973"> | ||
372 | </rect> | ||
373 | <g class="label" style="" | ||
374 | transform="translate(-22.725000381469727, -12.800000190734863)"> | ||
375 | <rect></rect> | ||
376 | <foreignObject width="45.45000076293945" height="25.600000381469727"> | ||
377 | <div xmlns="http://www.w3.org/1999/xhtml" | ||
378 | style="display: inline-block; white-space: nowrap;"><span | ||
379 | class="nodeLabel">Visual</span></div> | ||
380 | </foreignObject> | ||
381 | </g> | ||
382 | </g> | ||
383 | <g class="node default default flowchart-label" id="flowchart-R-29" | ||
384 | transform="translate(209.59375381469727, 201.49999618530273)"> | ||
385 | <rect class="basic label-container" style="" rx="5" ry="5" x="-37.23125076293945" | ||
386 | y="-20.300000190734863" width="74.4625015258789" height="40.60000038146973"> | ||
387 | </rect> | ||
388 | <g class="label" style="" | ||
389 | transform="translate(-29.731250762939453, -12.800000190734863)"> | ||
390 | <rect></rect> | ||
391 | <foreignObject width="59.462501525878906" height="25.600000381469727"> | ||
392 | <div xmlns="http://www.w3.org/1999/xhtml" | ||
393 | style="display: inline-block; white-space: nowrap;"><span | ||
394 | class="nodeLabel">Replace</span></div> | ||
395 | </foreignObject> | ||
396 | </g> | ||
397 | </g> | ||
398 | </g> | ||
399 | </g> | ||
400 | </g> | ||
401 | </svg></div> | ||
402 | <!-- 楚河汉界 --> | ||
403 | <h1 id="我的-vim-学习路线">我的 vim 学习路线</h1> | 61 | <h1 id="我的-vim-学习路线">我的 vim 学习路线</h1> |
404 | <p>不幸的是 Vim 的学习曲线确实相当陡峭,最开始你会觉得非常不适应,但一旦熬过了初始阶段,你会爱上 Vim。以我而言,现在离开了 vim 的模式几乎已经快不会写材料了,前几天交作业要求 | 62 | <p>不幸的是 Vim 的学习曲线确实相当陡峭,最开始你会觉得非常不适应,但一旦熬过了初始阶段,你会爱上 Vim。以我而言,现在离开了 vim 的模式几乎已经快不会写材料了,前几天交作业要求 |
405 | word,我保存文件居然不是<kbd>Ctrl+s</kbd>,而是顺手来了个<kbd>:wq</kbd>。</p> | 63 | word,我保存文件居然不是<kbd>Ctrl+s</kbd>,而是顺手来了个<kbd>:wq</kbd>。</p> |
@@ -437,8 +95,7 @@ | |||
437 | <li>不论何种 Linux 发行版,使用的 vim 版本都略微比发行时间旧一些以保证稳定性;但有时追求新功能的话需要最新版,甚至网上找不到你需要的版本的安装包。</li> | 95 | <li>不论何种 Linux 发行版,使用的 vim 版本都略微比发行时间旧一些以保证稳定性;但有时追求新功能的话需要最新版,甚至网上找不到你需要的版本的安装包。</li> |
438 | </ul> | 96 | </ul> |
439 | <p>因而,我们有时会需要自己手动编译安装 vim。即以我而言,为了在 vim 上使用 GitHub Copilot 插件,根据插件文档要求,我不得不手动编译 vim。下面是我的编译过程。当前系统为 | 97 | <p>因而,我们有时会需要自己手动编译安装 vim。即以我而言,为了在 vim 上使用 GitHub Copilot 插件,根据插件文档要求,我不得不手动编译 vim。下面是我的编译过程。当前系统为 |
440 | Ubuntu22.04 | 98 | Ubuntu22.04 LTS。编译过程中主要参考以下几篇文章,在此一并致谢:</p> |
441 | LTS。编译过程中主要参考以下几篇文章,在此一并致谢:</p> | ||
442 | <ul> | 99 | <ul> |
443 | <li><a href="https://richrose.dev/posts/linux/vim/vim-compile/">Compiling Vim</a></li> | 100 | <li><a href="https://richrose.dev/posts/linux/vim/vim-compile/">Compiling Vim</a></li> |
444 | <li><a href="https://blog.csdn.net/MHSMIE/article/details/112005119">vim 使用原码升级到最新版本</a></li> | 101 | <li><a href="https://blog.csdn.net/MHSMIE/article/details/112005119">vim 使用原码升级到最新版本</a></li> |
@@ -446,11 +103,11 @@ | |||
446 | </ul> | 103 | </ul> |
447 | <h2 id="清理干净">清理干净</h2> | 104 | <h2 id="清理干净">清理干净</h2> |
448 | <p>下载的第一步当然是卸载当前系统具有的 vim 了。除此之外,由于 Python2 过于老旧,我选择不支持 Python2 仅支持 Python3 的 vim,因此也将 Python2 卸载。</p> | 105 | <p>下载的第一步当然是卸载当前系统具有的 vim 了。除此之外,由于 Python2 过于老旧,我选择不支持 Python2 仅支持 Python3 的 vim,因此也将 Python2 卸载。</p> |
449 | <pre><code class="language-bash">sudo apt autoremove vim rvim gvim vim-runtime | 106 | <pre><code>sudo apt autoremove vim rvim gvim vim-runtime |
450 | sudo apt autoremove python2*</code></pre> | 107 | sudo apt autoremove python2*</code></pre> |
451 | <h2 id="做好准备">做好准备</h2> | 108 | <h2 id="做好准备">做好准备</h2> |
452 | <p>为了编译能够支持 python3/rust/perl/lua 等语言的 vim,我们需要安装一些包。</p> | 109 | <p>为了编译能够支持 python3/rust/perl/lua 等语言的 vim,我们需要安装一些包。</p> |
453 | <pre><code class="language-bash">sudo apt install -y git \ | 110 | <pre><code>sudo apt install -y git \ |
454 | libatk1.0-dev \ | 111 | libatk1.0-dev \ |
455 | libcairo2-dev \ | 112 | libcairo2-dev \ |
456 | #libgtk2.0-dev \ | 113 | #libgtk2.0-dev \ |
@@ -466,12 +123,11 @@ sudo apt autoremove python2*</code></pre> | |||
466 | <p>由于我的编译在服务器上运行,主要运行环境为命令行,不需要 gtk2 等图形界面的支持,因而将安装 gtk 的那一行注释掉了。在编译带有 gtk2.0 的 vim 时似乎会遇到问题:vim | 123 | <p>由于我的编译在服务器上运行,主要运行环境为命令行,不需要 gtk2 等图形界面的支持,因而将安装 gtk 的那一行注释掉了。在编译带有 gtk2.0 的 vim 时似乎会遇到问题:vim |
467 | 源码截至目前版本(v9.0.1854)采用的 gtk 代码中的<code>struct _GTimeVal</code>已经被弃用,gcc | 124 | 源码截至目前版本(v9.0.1854)采用的 gtk 代码中的<code>struct _GTimeVal</code>已经被弃用,gcc |
468 | 要求改用<code>struct _GDateTime</code>,因此编译时会报错,而且陷入死循环,编译不会终止,只能<kbd>Ctrl+c</kbd>杀死进程。我尚不知道如何解决。这也是我放弃支持 | 125 | 要求改用<code>struct _GDateTime</code>,因此编译时会报错,而且陷入死循环,编译不会终止,只能<kbd>Ctrl+c</kbd>杀死进程。我尚不知道如何解决。这也是我放弃支持 |
469 | gtk | 126 | gtk 的原因之一。</p> |
470 | 的原因之一。</p> | ||
471 | <h2 id="下载源码">下载源码</h2> | 127 | <h2 id="下载源码">下载源码</h2> |
472 | <p>vim 源码可以通过其<a href="https://github.com/vim/vim">GitHub 仓库</a>下载,也可以在其<a | 128 | <p>vim 源码可以通过其<a href="https://github.com/vim/vim">GitHub 仓库</a>下载,也可以在其<a |
473 | href="https://www.vim.org/download.php">官网</a>下载。我选择了前者。</p> | 129 | href="https://www.vim.org/download.php">官网</a>下载。我选择了前者。</p> |
474 | <pre><code class="language-bash">git clone https://github.com/vim/vim | 130 | <pre><code>git clone https://github.com/vim/vim |
475 | 131 | ||
476 | # 后续处理过程都在源码目录下的src目录内进行,因而切换过去 | 132 | # 后续处理过程都在源码目录下的src目录内进行,因而切换过去 |
477 | cd vim/src | 133 | cd vim/src |
@@ -483,7 +139,7 @@ sudo make distclean</code></pre> | |||
483 | <p>vim | 139 | <p>vim |
484 | 的编译过程中,需要先运行<code>configure</code>脚本,该脚本会检查当前系统的环境,并以参数形式接收你需要哪些功能。可以通过<code>./configure --help</code>查看所有可选参数及其功能说明。我选择的参数如下: | 140 | 的编译过程中,需要先运行<code>configure</code>脚本,该脚本会检查当前系统的环境,并以参数形式接收你需要哪些功能。可以通过<code>./configure --help</code>查看所有可选参数及其功能说明。我选择的参数如下: |
485 | </p> | 141 | </p> |
486 | <pre><code class="language-bash">sudo ./configure\ | 142 | <pre><code>sudo ./configure\ |
487 | --with-features=huge \ | 143 | --with-features=huge \ |
488 | --enable-multibyte \ | 144 | --enable-multibyte \ |
489 | --enable-rubyinterp=dynamic \ | 145 | --enable-rubyinterp=dynamic \ |
@@ -523,22 +179,22 @@ sudo make distclean</code></pre> | |||
523 | </p> | 179 | </p> |
524 | <h3 id="编译安装">编译安装</h3> | 180 | <h3 id="编译安装">编译安装</h3> |
525 | <p>编译安装过程简单到无以复加,就几句话的事。</p> | 181 | <p>编译安装过程简单到无以复加,就几句话的事。</p> |
526 | <pre><code class="language-bash"># 编译 | 182 | <pre><code># 编译 |
527 | sudo make | 183 | sudo make |
528 | # 安装 | 184 | # 安装 |
529 | sudo make install</code></pre> | 185 | sudo make install</code></pre> |
530 | <p>安装过程中可能会出现报错,如部分文件(<code>ex</code>/<code>view</code>等)已经存在,删掉该文件再重新运行命令安装便是。</p> | 186 | <p>安装过程中可能会出现报错,如部分文件(<code>ex</code>/<code>view</code>等)已经存在,删掉该文件再重新运行命令安装便是。</p> |
531 | <p>安装完成后,可以在命令行中运行<code>vim --version</code>查看 vim 的版本信息,确认是否支持你需要的功能。</p> | 187 | <p>安装完成后,可以在命令行中运行<code>vim --version</code>查看 vim 的版本信息,确认是否支持你需要的功能。</p> |
188 | <p>最后,将我的<a href="https://file.qin-juan-ge-zhu.top/compile_vim.sh">编译安装脚本</a>奉上,供参考。</p> | ||
532 | <h1 id="我的-vim-配置">我的 vim 配置</h1> | 189 | <h1 id="我的-vim-配置">我的 vim 配置</h1> |
533 | <p>网上一般的教程都说 vim 的默认配置文件为<code>~/.vimrc</code>。这当然是不错的,但是有一个巨大的问题:家目录是随着用户而不同的,这会导致我们以当前普通用户打开 vim 编辑文件与以 | 190 | <p>网上一般的教程都说 vim 的默认配置文件为<code>~/.vimrc</code>。这当然是不错的,但是有一个巨大的问题:家目录是随着用户而不同的,这会导致我们以当前普通用户打开 vim 编辑文件与以 |
534 | root | 191 | root 用户使用 vim 的观感完全不一致,就很烦。</p> |
535 | 用户使用 vim 的观感完全不一致,就很烦。</p> | ||
536 | <p>通过查看<kbd>vim –version</kbd>的输出,可以发现,vim 的配置文件有几层,其中“系统 vimrc | 192 | <p>通过查看<kbd>vim –version</kbd>的输出,可以发现,vim 的配置文件有几层,其中“系统 vimrc |
537 | 文件”被放置在<code>$VIM/vimrc</code>,而<code>$VIM</code>为<code>/etc/vim</code>。因此,我们可以将 vim | 193 | 文件”被放置在<code>$VIM/vimrc</code>,而<code>$VIM</code>为<code>/etc/vim</code>。因此,我们可以将 vim |
538 | 的配置文件放置在<code>/etc/vim</code>下,这样不论是普通用户还是 root | 194 | 的配置文件放置在<code>/etc/vim</code>下,这样不论是普通用户还是 root |
539 | 用户,都会使用这份文件,保证观感一致。另外,由于该文件位置较远,我选择在<code>~/useful</code>下创建软链接,方便使用。</p> | 195 | 用户,都会使用这份文件,保证观感一致。另外,由于该文件位置较远,我选择在<code>~/useful</code>下创建软链接,方便使用。</p> |
540 | <p>除此之外,出于同样的目的(保证当前用户与 root 用户观感一致),建议将插件也指定一个专门的绝对路径。我放在了<code>/home/player/useful/bundle</code>。</p> | 196 | <p>除此之外,出于同样的目的(保证当前用户与 root 用户观感一致),建议将插件也指定一个专门的绝对路径。我放在了<code>/home/player/useful/bundle</code>。</p> |
541 | <pre><code class="language-bash"># 创建软链接 | 197 | <pre><code># 创建软链接 |
542 | # 我习惯把有用的文件放在~/useful下 | 198 | # 我习惯把有用的文件放在~/useful下 |
543 | mkdir useful | 199 | mkdir useful |
544 | cd useful | 200 | cd useful |
@@ -549,377 +205,24 @@ sudo ln -s /etc/vim /home/player/useful/vim | |||
549 | # 插件存放位置 | 205 | # 插件存放位置 |
550 | mkdir bundle</code></pre> | 206 | mkdir bundle</code></pre> |
551 | <h2 id="设置">设置</h2> | 207 | <h2 id="设置">设置</h2> |
552 | <p>下文所有设置的目录均为<code>~/useful/vim</code>,也即<code>/etc/vim</code>。由于内容过多,我在目录下创建了<code>vimrcs</code>文件夹,将不同内容分别以专门文件管理,vimrc | 208 | <p>这里原本记录着我的旧版 vim 配置,但随着<a href="https://www.qin-juan-ge-zhu.top/code/linux/httpmyserver.html">http |
553 | 文件调用这些文件。因此,下边只有<code>vimrc</code>在<code>/etc/vim</code>下;其他文件均在<code>/etc/vim/vimrcs</code>下,以<code>./vimrcs/<文件名></code>表示。插件放置在<code>~/useful/bundle</code>。 | 209 | 浏览服务器文件</a>的完成,再保留旧版然后每隔三四个月更新一次,对于我这种没事就折腾的人来说说,就显得不那么必要了。因此,这里只简单说一下。</p> |
554 | </p> | 210 | <p>配置的整体结构如下:</p> |
555 | <h3 id="vimrc">./vimrc</h3> | 211 | <pre><code>/etc/vim |
556 | <pre><code class="hljs language-vim">" Vundle插件管理及插件内容 | 212 | ├── vimrc # vim配置文件,负责引入不同的配置模块 |
557 | if filereadable(expand("/etc/vim/vimrcs/plugs.vim")) | 213 | ├── vimrc_origin # 原有配置脚本备份 |
558 | source /etc/vim/vimrcs/plugs.vim | 214 | ├── vimrcs # 我的配置文件夹,里边是不同的配置模块 |
559 | endif | 215 | │ ├── autocmd.vim |
560 | 216 | │ ├── codecmd.vim | |
561 | " 自行设置 | 217 | │ ├── keybind.vim |
562 | if filereadable(expand("/etc/vim/vimrcs/myset.vim")) | 218 | │ ├── match.vim |
563 | source /etc/vim/vimrcs/myset.vim | 219 | │ ├── myset.vim |
564 | endif | 220 | │ ├── plugs.vim |
565 | 221 | │ └── statusline.vim | |
566 | " 符号自动匹配 | 222 | ├── vimrc.tiny # 原有脚本,无用 |
567 | if filereadable(expand("/etc/vim/vimrcs/match.vim")) | 223 | └── vimscript.vim # 原有脚本,无用</code></pre> |
568 | source /etc/vim/vimrcs/match.vim | 224 | <p>我直接把我服务器上的配置文件<a href="https://file.qin-juan-ge-zhu.top/useful/vim/">放在这里</a>,以供查看。</p> |
569 | endif | 225 | <p class="time">2023.7.5<br>2023.9.6<br>2024.1.3</p> |
570 | |||
571 | " 不知道是什么 | ||
572 | if filereadable(expand("/etc/vim/vimrcs/fuck.vim")) | ||
573 | source /etc/vim/vimrcs/fuck.vim | ||
574 | endif | ||
575 | |||
576 | " 自己设的快捷键 | ||
577 | if filereadable(expand("/etc/vim/vimrcs/keybind.vim")) | ||
578 | source /etc/vim/vimrcs/keybind.vim | ||
579 | endif | ||
580 | |||
581 | " 自动命令 | ||
582 | if filereadable(expand("/etc/vim/vimrcs/autocmd.vim")) | ||
583 | source /etc/vim/vimrcs/autocmd.vim | ||
584 | endif</code></pre> | ||
585 | <h3 id="vimrcsplugs.vim">./vimrcs/plugs.vim</h3> | ||
586 | <pre><code class="hljs language-vim">" Vundle插件管理器配置 | ||
587 | set nocompatible "去除VIM一致性,必须 | ||
588 | filetype off "必须 | ||
589 | |||
590 | "设置包括vundle和初始化相关的运行时路径 | ||
591 | set rtp+=/home/player/useful/bundle/Vundle.vim | ||
592 | call vundle#begin('/home/player/useful/bundle') | ||
593 | |||
594 | Plugin 'VundleVim/Vundle.vim' "启用vundle管理插件,必须 | ||
595 | Plugin 'Valloric/YouCompleteMe' "代码补全插件 | ||
596 | Plugin 'preservim/nerdtree' "文件目录树插件 | ||
597 | Plugin 'Xuyuanp/nerdtree-git-plugin' "nerdtree git支持 | ||
598 | Plugin 'skywind3000/vim-terminal-help' "vim-terminal-help插件 | ||
599 | Plugin 'scrooloose/syntastic' "语法检查插件 | ||
600 | Plugin 'Lokaltog/powerline',{'rtp':'powerline/bindings/vim/'} "状态栏信息插件 | ||
601 | Plugin 'ianva/vim-youdao-translater' "有道翻译插件 | ||
602 | Plugin 'github/copilot.vim' "GitHub Copilot插件 | ||
603 | |||
604 | "在此增加其他插件,安装的插件需要放在vundle#begin和vundle#end之间 | ||
605 | "安装github上的插件格式为 Plugin '用户名/插件仓库名' | ||
606 | |||
607 | call vundle#end() | ||
608 | |||
609 | filetype plugin indent on "加载vim自带和插件相应的语法和文件类型相关脚本,必须 | ||
610 | |||
611 | " YouCompleteMe | ||
612 | let g:ycm_global_ycm_extra_conf='/home/player/useful/bundle/YouCompleteMe/.ycm_extra_conf.py' | ||
613 | let g:ycm_key_invoke_completion = '<C-c>' | ||
614 | let g:ycm_semantic_triggers = { | ||
615 | \ 'c,cpp,python,java,go,erlang,perl': ['re!\w{2}'], | ||
616 | \ 'cs,lua,javascript': ['re!\w{2}'], | ||
617 | \ } | ||
618 | highlight PMenu ctermfg=34 ctermbg=21 guifg=darkred guibg=darkblue | ||
619 | highlight PMenuSel ctermfg=21 ctermbg=34 guifg=darkblue guibg=darkred | ||
620 | let g:ycm_autoclose_preview_window_after_completion=1 "窗口在完成操作后消失 | ||
621 | map <leader>g :YcmCompleter GoToDefinitionElseDeclaration<CR> "找到当前内容定义位置的快捷方式 | ||
622 | |||
623 | "NERDTree | ||
624 | autocmd vimenter * NERDTree "进入vim自动打开nerdtree | ||
625 | autocmd bufenter * if(winnr("$") == 1 && exists("b:NERDTree") && b:NERDTree.isTabTree()) | q | endif "当nerdtree为剩余唯一窗口时自动关闭 | ||
626 | map <F2> :NERDTreeToggle<CR> "F2键开关文件目录树,注意shift+i切换是否显示隐藏文件 | ||
627 | let g:NERDTreeDirArrowExpandable = '+' | ||
628 | let g:NERDTreeDirArrowCollapsible = '-' "修改树的显示图标 | ||
629 | "let g:NERDTreeWinPos='left' "窗口位置 | ||
630 | let g:NERDTreeSize=10"窗口尺寸 | ||
631 | let g:NERDTreeShowLineNumbers=1 "窗口是否显示行号 | ||
632 | "let g:NERDTreeHidden=1 "似乎是隐藏文件的旧命令?没看到作用 | ||
633 | let NERDTreeShowHidden=1 " 显示隐藏文件 | ||
634 | let NERDTreeIgnore = ['\.pyc$', '\.swp', '\.swo', '\.vscode', '__pycache__'] " 过滤: 所有指定文件和文件夹不显示 | ||
635 | autocmd BufWinEnter * if getcmdwintype() == '' | silent NERDTreeMirror | endif " 在每个标签页打开相同的文件树 | ||
636 | let g:NERDtreeNodeDelimiter = "\u00a0" | ||
637 | |||
638 | |||
639 | "nerdtree-git-plugin | ||
640 | let g:NERDTreeGitStatusIndicatorMapCustom = { | ||
641 | \ 'Dirty' :'Dir', | ||
642 | \ 'Modified' :'M', | ||
643 | \ 'Staged' :'S', | ||
644 | \ 'Untracked' :'U', | ||
645 | \ 'Renamed' :'R', | ||
646 | \ 'Unmerged' :'UM', | ||
647 | \ 'Deleted' :'X', | ||
648 | \ 'Ignored' :'Ig', | ||
649 | \ 'Clean' :'Cl', | ||
650 | \ 'Unknown' :'?', | ||
651 | \ } | ||
652 | let g:NERDTreeGitStatusShowIgnored = 1 " a heavy feature may cost much more time. default: 0 | ||
653 | let g:NERDTreeGitStatusUntrackedFilesMode = 'all' " a heavy feature too. default: normal | ||
654 | "let g:NERDTreeGitStatusGitBinPath = '/your/file/path' " default: git (auto find in path) | ||
655 | |||
656 | "vim-terminal-help | ||
657 | set termwinsize=10*106 | ||
658 | |||
659 | "syntastic | ||
660 | "以下内容是syntastic插件的一般配置,由于YouCompleteMe插件对该插件支持过于优秀,一般不再需要配置,故废弃 | ||
661 | "set statusline+=%#warningmsg# | ||
662 | "set statusline+=%{syntasticstatuslineflag()} | ||
663 | "set statusline+=%* | ||
664 | " | ||
665 | "let g:syntastic_always_populate_loc_list = 1 | ||
666 | "let g:syntastic_auto_loc_list = 1 | ||
667 | "let g:syntastic_check_on_open = 1 | ||
668 | "let g:syntastic_check_on_wq = 0 | ||
669 | ""设置error和warning的标志 | ||
670 | "let g:syntastic_enable_signs = 1 | ||
671 | ""let g:syntastic_error_symbol='?' | ||
672 | "let g:syntastic_warning_symbol='?' | ||
673 | ""no-unused-args 忽略不使用的参数 | ||
674 | ""no-redefined 忽略重定义 | ||
675 | ""no-max-line-length 忽略每行最长的检测 | ||
676 | ""ignore 542 551 忽略if条件的body为空(如...elseif ret == nil then end,then和end中间没有语句);忽略空语句(如;),如果用了;则成对出现该语法检测warning | ||
677 | ""有语法错误的一行左侧栏会有s&gt;标识,光标移动到改行,vim下发会给出提示。修改正确后保存,则该's&gt;'会消失。 | ||
678 | "let g:syntastic_lua_checkers = ["/usr/bin/luac5.3.5", "luacheck"] | ||
679 | "let g:syntastic_lua_luacheck_args = "--codes --no-max-comment-line-length --ignore 542 551" | ||
680 | " | ||
681 | ""c | ||
682 | ""let g:syntastic_c_compiler =['gcc', 'clang', 'make'] | ||
683 | ""let g:syntastic_c_compiler_options ='-wpedantic -g' | ||
684 | "let g:syntastic_c_compiler_options ='-std=gnu99' | ||
685 | "let g:syntastic_c_include_dirs=['/usr/include/'] | ||
686 | "let g:syntastic_c_config_file='.syntastic_c_config_file' | ||
687 | " | ||
688 | ""cpp | ||
689 | "let g:syntastic_cpp_include_dirs = ['/usr/include/'] | ||
690 | "let g:syntastic_cpp_remove_include_errors = 1 | ||
691 | "let g:syntastic_cpp_check_header = 1 | ||
692 | "let g:syntastic_cpp_compiler = 'clang++' | ||
693 | "let g:syntastic_cpp_compiler_options = '-std=c++11 -stdlib=libstdc++' | ||
694 | "" syntastic end | ||
695 | let g:syntastic_python_checkers=['pylint'] | ||
696 | let g:syntastic_python_pylint_args=['--disable=C0111,R0903,C0301'] | ||
697 | |||
698 | |||
699 | " powerline | ||
700 | let laststatus=2 | ||
701 | let g:airlinr_powerline_fonts=1 "使用官方打过补丁的字体 | ||
702 | let g:Powerline_symbols='fancy' | ||
703 | |||
704 | "vim-youdao-translater | ||
705 | vnoremap <silent> <C-t> :<C-u>Ydv<CR> | ||
706 | nnoremap <silent> <C-t> :<C-u>Ydc<CR> | ||
707 | noremap <leader>yd: <C-u>Yde<CR></code></pre> | ||
708 | <h3 id="vimrcsmyset.vim">./vimrcs/myset.vim</h3> | ||
709 | <pre><code class="hljs language-vim">set nocp "设置兼容 | ||
710 | set expandtab "设置tab | ||
711 | set shiftwidth=4 "设置tab的间隔 | ||
712 | set tabstop=4 "四个空格代表一个tab | ||
713 | set sts=4 | ||
714 | set showmatch "在输入括号时光标会短暂地跳到与之相匹配的括号处 | ||
715 | set autoindent "设置自动缩进 | ||
716 | " set smartindent "设置智能缩进 | ||
717 | set nowrap "设置自动换行 | ||
718 | set tw=500 | ||
719 | set lbr | ||
720 | set foldmethod=indent "设置按缩进折叠代码 | ||
721 | set foldlevel=99 "折叠层级 | ||
722 | nnoremap <space> za "空格代替za进行展开和折叠 | ||
723 | "===set guifont=Monospace "设置字体大小 | ||
724 | set guifont=:b:h16 | ||
725 | set encoding=utf-8 "设置编码为utf-8 | ||
726 | set fileencoding=utf-8 | ||
727 | set fileencodings=ucs-bom,utf-8,GB18030,cp936,big5,euc-jp,euc-kr,latin1 | ||
728 | set helplang=cn "帮助中文支持 | ||
729 | "colorscheme evening "设置主体颜色 | ||
730 | set mouse=a " 设置鼠标 | ||
731 | |||
732 | "自动补全配置 | ||
733 | autocmd FileType python set omnifunc=pythoncomplete#Complete | ||
734 | autocmd FileType javascript set omnifunc=javascriptcomplete#CompleteJS | ||
735 | autocmd FileType html set omnifunc=htmlcomplete#CompleteTags | ||
736 | autocmd FileType css set omnifunc=csscomplete#CompleteCSS | ||
737 | autocmd FileType xml set omnifunc=xmlcomplete#CompleteTags | ||
738 | autocmd FileType php set omnifunc=phpcomplete#CompletePHP | ||
739 | autocmd FileType c set omnifunc=ccomplete#Complete | ||
740 | |||
741 | "进行Taglist的设置 | ||
742 | map <F3> :TlistToggle<CR> | ||
743 | "map <F3> :silent! Tlist<CR> "按下F3就可以呼出了 | ||
744 | "let Tlist_Ctags_Cmd='/usr/bin/ctags' "因为我们放在环境变量里,所以可以直接执行 | ||
745 | let Tlist_Use_Right_Window=1 "让窗口显示在右边,0的话就是显示在左边 | ||
746 | "let Tlist_Show_One_File=1 "让taglist可以同时展示多个文件的函数列表 | ||
747 | "let Tlist_File_Fold_Auto_Close=1 "非当前文件,函数列表折叠隐藏 | ||
748 | "let Tlist_Exit_OnlyWindow=1 "当taglist是最后一个分割窗口时,自动推出vim | ||
749 | "let Tlist_Process_File_Always=0 "是否一直处理tags.1:处理;0:不处理 | ||
750 | "let Tlist_Inc_Winwidth=0 "不是一直实时更新tags,因为没有必要 | ||
751 | |||
752 | set rnu "设置相对行号 | ||
753 | set nu "设置绝对行号 | ||
754 | " Set cursor shape and color | ||
755 | if &term =~ "xterm" | ||
756 | " INSERT mode | ||
757 | let &t_SI = "\<Esc>[6 q" . "\<Esc>]12;blue\x7" | ||
758 | " REPLACE mode | ||
759 | let &t_SR = "\<Esc>[3 q" . "\<Esc>]12;black\x7" | ||
760 | " NORMAL mode | ||
761 | let &t_EI = "\<Esc>[2 q" . "\<Esc>]12;green\x7" | ||
762 | endif | ||
763 | " 1 -> blinking block 闪烁的方块 | ||
764 | " 2 -> solid block 不闪烁的方块 | ||
765 | " 3 -> blinking underscore 闪烁的下划线 | ||
766 | " 4 -> solid underscore 不闪烁的下划线 | ||
767 | " 5 -> blinking vertical bar 闪烁的竖线 | ||
768 | " 6 -> solid vertical bar 不闪烁的竖线 | ||
769 | |||
770 | syntax on | ||
771 | set cul | ||
772 | set cuc | ||
773 | hi CursorLine cterm=NONE ctermbg=darkred ctermfg=white guibg=darkred guifg=white | ||
774 | ""set lines=35 columns=118 | ||
775 | colorscheme zaibatsu "设置颜色主题 | ||
776 | set wrap "设置自动折行</code></pre> | ||
777 | <h3 id="vimrcsmatch.vim">./vimrcs/match.vim</h3> | ||
778 | <pre><code class="hljs language-vim">"各类符号自动匹配 | ||
779 | :inoremap ( ()<ESC>i | ||
780 | :inoremap ) <c-r>=ClosePair(')')<CR> | ||
781 | :inoremap { {}<ESC>i | ||
782 | :inoremap } <c-r>=ClosePair('}')<CR> | ||
783 | :inoremap [ []<ESC>i | ||
784 | :inoremap ] <c-r>=ClosePair(']')<CR> | ||
785 | |||
786 | function ClosePair(char) | ||
787 | if getline('.')[col('.') - 1] == a:char | ||
788 | return "<Right>" | ||
789 | else | ||
790 | return a:char | ||
791 | endif | ||
792 | endfunction</code></pre> | ||
793 | <h3 id="vimrcsfuck.vim">./vimrcs/fuck.vim</h3> | ||
794 | <pre><code class="hljs language-vim">" 键盘映射设置区域 | ||
795 | |||
796 | " 对应使用函数的配置 | ||
797 | set guitablabel=%{ShortTabLabel()} | ||
798 | function ShortTabLabel () | ||
799 | let bufnrlist = tabpagebuflist (v:lnum) | ||
800 | let label = bufname (bufnrlist[tabpagewinnr (v:lnum) -1]) | ||
801 | let filename = fnamemodify (label, ':t') | ||
802 | return filename | ||
803 | endfunction | ||
804 | |||
805 | set tabline=%!MyTabLine() | ||
806 | function MyTabLine() | ||
807 | let s = '' | ||
808 | for i in range(tabpagenr('$')) | ||
809 | " 选择高亮 | ||
810 | if i + 1 == tabpagenr() | ||
811 | let s .= '%#TabLineSel#' | ||
812 | else | ||
813 | let s .= '%#TabLine#' | ||
814 | endif | ||
815 | " 设置标签页号 (用于鼠标点击) | ||
816 | let s .= '%' . (i + 1) . 'T' | ||
817 | " MyTabLabel() 提供完整路径标签 MyShortTabLabel 提供文件名标签 | ||
818 | let s .= ' %{MyShortTabLabel(' . (i + 1) . ')} ' | ||
819 | endfor | ||
820 | " 最后一个标签页之后用 TabLineFill 填充并复位标签页号 | ||
821 | let s .= '%#TabLineFill#%T' | ||
822 | " 右对齐用于关闭当前标签页的标签 | ||
823 | if tabpagenr('$') > 1 | ||
824 | let s .= '%=%#TabLine#%999Xclose' | ||
825 | endif | ||
826 | return s | ||
827 | endfunction | ||
828 | " 文件名标签 | ||
829 | function MyShortTabLabel(n) | ||
830 | let buflist = tabpagebuflist(a:n) | ||
831 | let label = bufname (buflist[tabpagewinnr (a:n) -1]) | ||
832 | let filename = fnamemodify (label, ':t') | ||
833 | return filename | ||
834 | endfunction | ||
835 | "完整路径标签 | ||
836 | function MyTabLabel(n) | ||
837 | let buflist = tabpagebuflist(a:n) | ||
838 | let winnr = tabpagewinnr(a:n) | ||
839 | return bufname(buflist[winnr - 1]) | ||
840 | endfunction | ||
841 | " vim 标签样式 | ||
842 | " TabLineFill tab pages line, where there are no labels | ||
843 | hi TabLineFill term=none | ||
844 | hi TabLineFill ctermfg=DarkGrey | ||
845 | hi TabLineFill guifg=#777777 | ||
846 | " TabLineSel tab pages line, active tab page label | ||
847 | hi TabLineSel term=inverse | ||
848 | hi TabLineSel cterm=none ctermfg=yellow ctermbg=Black | ||
849 | hi TabLineSel gui=none guifg=yellow guibg=Black | ||
850 | |||
851 | " Develop editing options | ||
852 | au FileType vim setl expandtab | ||
853 | au FileType vim setl shiftwidth=2 | ||
854 | au FileType vim setl tabstop=2 | ||
855 | |||
856 | " 显示状态栏(默认值为 1,无法显示状态栏) | ||
857 | set laststatus=2 | ||
858 | " Format the statusline | ||
859 | " set statusline= %f %m %r %h %w CWD: %r%{CurDir()}E5%h Line: %l/%L:%c | ||
860 | |||
861 | function! CurDir() | ||
862 | let curdir = substitute(getcwd(), '/Users/amir/', "~/", "g") | ||
863 | return curdir | ||
864 | endfunction</code></pre> | ||
865 | <h3 id="vimrcskeybind.vim">./vimrcs/keybind.vim</h3> | ||
866 | <pre><code class="hljs language-vim">map - dd | ||
867 | map up 20k | ||
868 | map ne 20j | ||
869 | map end G | ||
870 | map ta $ | ||
871 | "map <C-l> <C-w-l> | ||
872 | nmap <C-tab> :tabn<CR> | ||
873 | nmap <tab> :tabp<CR> | ||
874 | "nmap <alt-tab> :tabnew<CR> | ||
875 | map <space>= <buffer><esc>:vertical resize +5<cr> | ||
876 | map <space>- <esc>:vertical resize -5<cr> | ||
877 | |||
878 | nn <M-1> 1gt | ||
879 | nn <M-2> 2gt | ||
880 | nn <M-3> 3gt | ||
881 | nn <M-4> 4gt | ||
882 | nn <M-5> 5gt | ||
883 | nn <M-6> 6gt | ||
884 | nn <M-7> 7gt | ||
885 | nn <M-8> 8gt | ||
886 | nn <M-9> 9gt | ||
887 | nn <M-0> :tablast<CR> | ||
888 | |||
889 | "光标在页面间切换 | ||
890 | nnoremap <space>h <C-w><C-h> | ||
891 | nnoremap <space>j <C-w><C-j> | ||
892 | nnoremap <space>k <C-w><C-k> | ||
893 | nnoremap <space>l <C-w><C-l> | ||
894 | |||
895 | "复制粘贴 | ||
896 | "vnoremap <C-y> "+y | ||
897 | "nnoremap <C-p> "+p | ||
898 | |||
899 | " 其他开发配置 | ||
900 | " Python | ||
901 | au BufNewFile,BufRead *.py | ||
902 | \ set tabstop=4 | | ||
903 | \ set shiftwidth=4 | | ||
904 | \ set textwidth=79 | | ||
905 | \ set expandtab | | ||
906 | \ set autoindent | | ||
907 | \ set fileformat=unix |</code></pre> | ||
908 | <h3 id="vimrcsautocmd.vim">./vimrcs/autocmd.vim</h3> | ||
909 | <pre><code class="hljs language-vim">autocmd BufReadPost,BufWritePre *.html,*.c,*.cpp normal gg=G | ||
910 | autocmd Filetype html,markdown setlocal nowrap | ||
911 | "autocmd Filetype c,cpp map <buffer><C-p> I//<esc> | ||
912 | autocmd Filetype c,cpp map <buffer><C-d> 0xx<esc>ta | ||
913 | "autocmd Filetype c,cpp vmap <buffer><C-p> <C-v>I//<esc> | ||
914 | "autocmd Filetype python map <buffer><C-p> I# <esc> | ||
915 | autocmd Filetype python map <buffer><C-d> 0xx<esc>ta | ||
916 | "autocmd Filetype python vmap <buffer><C-p> <C-v>I# <esc> | ||
917 | "autocmd Filetype c,cpp vmap <buffer><C-d> <C-v>0xx | ||
918 | "说明:本文件上文所有<Ctrl-p>的设置,原是用来为c,cpp,python文件设置注释的 | ||
919 | "但是注释可以通过<Ctrl-v>进入块选模式而后<Ctrl-i>进入插入模式 | ||
920 | "在前边输入注释符号,保存,则选中的行均被注释 | ||
921 | "甚为方便,因此这许多键一起弃用。</code></pre> | ||
922 | <p class="time">2023.7.5<br>2023.9.6改</p> | ||
923 | <script src="https://www.qin-juan-ge-zhu.top/common/js/comment.js"></script> | 226 | <script src="https://www.qin-juan-ge-zhu.top/common/js/comment.js"></script> |
924 | </div> | 227 | </div> |
925 | </div> | 228 | </div> |
diff --git a/code/linux/vim.md b/code/linux/vim.md index 51b9263..ec32d67 100644 --- a/code/linux/vim.md +++ b/code/linux/vim.md | |||
@@ -159,6 +159,8 @@ sudo make install | |||
159 | 159 | ||
160 | 安装完成后,可以在命令行中运行`vim --version`查看 vim 的版本信息,确认是否支持你需要的功能。 | 160 | 安装完成后,可以在命令行中运行`vim --version`查看 vim 的版本信息,确认是否支持你需要的功能。 |
161 | 161 | ||
162 | 最后,将我的[编译安装脚本](https://file.qin-juan-ge-zhu.top/compile_vim.sh)奉上,供参考。 | ||
163 | |||
162 | # 我的 vim 配置 | 164 | # 我的 vim 配置 |
163 | 165 | ||
164 | 网上一般的教程都说 vim 的默认配置文件为`~/.vimrc`。这当然是不错的,但是有一个巨大的问题:家目录是随着用户而不同的,这会导致我们以当前普通用户打开 vim 编辑文件与以 root 用户使用 vim 的观感完全不一致,就很烦。 | 166 | 网上一般的教程都说 vim 的默认配置文件为`~/.vimrc`。这当然是不错的,但是有一个巨大的问题:家目录是随着用户而不同的,这会导致我们以当前普通用户打开 vim 编辑文件与以 root 用户使用 vim 的观感完全不一致,就很烦。 |
@@ -182,397 +184,24 @@ mkdir bundle | |||
182 | 184 | ||
183 | ## 设置 | 185 | ## 设置 |
184 | 186 | ||
185 | 下文所有设置的目录均为`~/useful/vim`,也即`/etc/vim`。由于内容过多,我在目录下创建了`vimrcs`文件夹,将不同内容分别以专门文件管理,vimrc 文件调用这些文件。因此,下边只有`vimrc`在`/etc/vim`下;其他文件均在`/etc/vim/vimrcs`下,以`./vimrcs/<文件名>`表示。插件放置在`~/useful/bundle`。 | 187 | 这里原本记录着我的旧版 vim 配置,但随着[http 浏览服务器文件](https://www.qin-juan-ge-zhu.top/code/linux/httpmyserver.html)的完成,再保留旧版然后每隔三四个月更新一次,对于我这种没事就折腾的人来说说,就显得不那么必要了。因此,这里只简单说一下。 |
186 | 188 | ||
187 | ### ./vimrc | 189 | 配置的整体结构如下: |
188 | 190 | ||
189 | ```vim | 191 | ```plaintext |
190 | " Vundle插件管理及插件内容 | 192 | /etc/vim |
191 | if filereadable(expand("/etc/vim/vimrcs/plugs.vim")) | 193 | ├── vimrc # vim配置文件,负责引入不同的配置模块 |
192 | source /etc/vim/vimrcs/plugs.vim | 194 | ├── vimrc_origin # 原有配置脚本备份 |
193 | endif | 195 | ├── vimrcs # 我的配置文件夹,里边是不同的配置模块 |
194 | 196 | │ ├── autocmd.vim | |
195 | " 自行设置 | 197 | │ ├── codecmd.vim |
196 | if filereadable(expand("/etc/vim/vimrcs/myset.vim")) | 198 | │ ├── keybind.vim |
197 | source /etc/vim/vimrcs/myset.vim | 199 | │ ├── match.vim |
198 | endif | 200 | │ ├── myset.vim |
199 | 201 | │ ├── plugs.vim | |
200 | " 符号自动匹配 | 202 | │ └── statusline.vim |
201 | if filereadable(expand("/etc/vim/vimrcs/match.vim")) | 203 | ├── vimrc.tiny # 原有脚本,无用 |
202 | source /etc/vim/vimrcs/match.vim | 204 | └── vimscript.vim # 原有脚本,无用 |
203 | endif | ||
204 | |||
205 | " 不知道是什么 | ||
206 | if filereadable(expand("/etc/vim/vimrcs/fuck.vim")) | ||
207 | source /etc/vim/vimrcs/fuck.vim | ||
208 | endif | ||
209 | |||
210 | " 自己设的快捷键 | ||
211 | if filereadable(expand("/etc/vim/vimrcs/keybind.vim")) | ||
212 | source /etc/vim/vimrcs/keybind.vim | ||
213 | endif | ||
214 | |||
215 | " 自动命令 | ||
216 | if filereadable(expand("/etc/vim/vimrcs/autocmd.vim")) | ||
217 | source /etc/vim/vimrcs/autocmd.vim | ||
218 | endif | ||
219 | ``` | ||
220 | |||
221 | ### ./vimrcs/plugs.vim | ||
222 | |||
223 | ```vim" Vundle插件管理器配置 | ||
224 | set nocompatible "去除VIM一致性,必须 | ||
225 | filetype off "必须 | ||
226 | |||
227 | "设置包括vundle和初始化相关的运行时路径 | ||
228 | set rtp+=/home/player/useful/bundle/Vundle.vim | ||
229 | call vundle#begin('/home/player/useful/bundle') | ||
230 | |||
231 | Plugin 'VundleVim/Vundle.vim' "启用vundle管理插件,必须 | ||
232 | Plugin 'Valloric/YouCompleteMe' "代码补全插件 | ||
233 | Plugin 'preservim/nerdtree' "文件目录树插件 | ||
234 | Plugin 'Xuyuanp/nerdtree-git-plugin' "nerdtree git支持 | ||
235 | Plugin 'skywind3000/vim-terminal-help' "vim-terminal-help插件 | ||
236 | Plugin 'scrooloose/syntastic' "语法检查插件 | ||
237 | Plugin 'Lokaltog/powerline',{'rtp':'powerline/bindings/vim/'} "状态栏信息插件 | ||
238 | Plugin 'ianva/vim-youdao-translater' "有道翻译插件 | ||
239 | Plugin 'github/copilot.vim' "GitHub Copilot插件 | ||
240 | |||
241 | "在此增加其他插件,安装的插件需要放在vundle#begin和vundle#end之间 | ||
242 | "安装github上的插件格式为 Plugin '用户名/插件仓库名' | ||
243 | |||
244 | call vundle#end() | ||
245 | |||
246 | filetype plugin indent on "加载vim自带和插件相应的语法和文件类型相关脚本,必须 | ||
247 | |||
248 | " YouCompleteMe | ||
249 | let g:ycm_global_ycm_extra_conf='/home/player/useful/bundle/YouCompleteMe/.ycm_extra_conf.py' | ||
250 | let g:ycm_key_invoke_completion = '<C-c>' | ||
251 | let g:ycm_semantic_triggers = { | ||
252 | \ 'c,cpp,python,java,go,erlang,perl': ['re!\w{2}'], | ||
253 | \ 'cs,lua,javascript': ['re!\w{2}'], | ||
254 | \ } | ||
255 | highlight PMenu ctermfg=34 ctermbg=21 guifg=darkred guibg=darkblue | ||
256 | highlight PMenuSel ctermfg=21 ctermbg=34 guifg=darkblue guibg=darkred | ||
257 | let g:ycm_autoclose_preview_window_after_completion=1 "窗口在完成操作后消失 | ||
258 | map <leader>g :YcmCompleter GoToDefinitionElseDeclaration<CR> "找到当前内容定义位置的快捷方式 | ||
259 | |||
260 | "NERDTree | ||
261 | autocmd vimenter * NERDTree "进入vim自动打开nerdtree | ||
262 | autocmd bufenter * if(winnr("$") == 1 && exists("b:NERDTree") && b:NERDTree.isTabTree()) | q | endif "当nerdtree为剩余唯一窗口时自动关闭 | ||
263 | map <F2> :NERDTreeToggle<CR> "F2键开关文件目录树,注意shift+i切换是否显示隐藏文件 | ||
264 | let g:NERDTreeDirArrowExpandable = '+' | ||
265 | let g:NERDTreeDirArrowCollapsible = '-' "修改树的显示图标 | ||
266 | "let g:NERDTreeWinPos='left' "窗口位置 | ||
267 | let g:NERDTreeSize=10"窗口尺寸 | ||
268 | let g:NERDTreeShowLineNumbers=1 "窗口是否显示行号 | ||
269 | "let g:NERDTreeHidden=1 "似乎是隐藏文件的旧命令?没看到作用 | ||
270 | let NERDTreeShowHidden=1 " 显示隐藏文件 | ||
271 | let NERDTreeIgnore = ['\.pyc$', '\.swp', '\.swo', '\.vscode', '__pycache__'] " 过滤: 所有指定文件和文件夹不显示 | ||
272 | autocmd BufWinEnter * if getcmdwintype() == '' | silent NERDTreeMirror | endif " 在每个标签页打开相同的文件树 | ||
273 | let g:NERDtreeNodeDelimiter = "\u00a0" | ||
274 | |||
275 | |||
276 | "nerdtree-git-plugin | ||
277 | let g:NERDTreeGitStatusIndicatorMapCustom = { | ||
278 | \ 'Dirty' :'Dir', | ||
279 | \ 'Modified' :'M', | ||
280 | \ 'Staged' :'S', | ||
281 | \ 'Untracked' :'U', | ||
282 | \ 'Renamed' :'R', | ||
283 | \ 'Unmerged' :'UM', | ||
284 | \ 'Deleted' :'X', | ||
285 | \ 'Ignored' :'Ig', | ||
286 | \ 'Clean' :'Cl', | ||
287 | \ 'Unknown' :'?', | ||
288 | \ } | ||
289 | let g:NERDTreeGitStatusShowIgnored = 1 " a heavy feature may cost much more time. default: 0 | ||
290 | let g:NERDTreeGitStatusUntrackedFilesMode = 'all' " a heavy feature too. default: normal | ||
291 | "let g:NERDTreeGitStatusGitBinPath = '/your/file/path' " default: git (auto find in path) | ||
292 | |||
293 | "vim-terminal-help | ||
294 | set termwinsize=10*106 | ||
295 | |||
296 | "syntastic | ||
297 | "以下内容是syntastic插件的一般配置,由于YouCompleteMe插件对该插件支持过于优秀,一般不再需要配置,故废弃 | ||
298 | "set statusline+=%#warningmsg# | ||
299 | "set statusline+=%{syntasticstatuslineflag()} | ||
300 | "set statusline+=%* | ||
301 | " | ||
302 | "let g:syntastic_always_populate_loc_list = 1 | ||
303 | "let g:syntastic_auto_loc_list = 1 | ||
304 | "let g:syntastic_check_on_open = 1 | ||
305 | "let g:syntastic_check_on_wq = 0 | ||
306 | ""设置error和warning的标志 | ||
307 | "let g:syntastic_enable_signs = 1 | ||
308 | ""let g:syntastic_error_symbol='?' | ||
309 | "let g:syntastic_warning_symbol='?' | ||
310 | ""no-unused-args 忽略不使用的参数 | ||
311 | ""no-redefined 忽略重定义 | ||
312 | ""no-max-line-length 忽略每行最长的检测 | ||
313 | ""ignore 542 551 忽略if条件的body为空(如...elseif ret == nil then end,then和end中间没有语句);忽略空语句(如;),如果用了;则成对出现该语法检测warning | ||
314 | ""有语法错误的一行左侧栏会有s>标识,光标移动到改行,vim下发会给出提示。修改正确后保存,则该's>'会消失。 | ||
315 | "let g:syntastic_lua_checkers = ["/usr/bin/luac5.3.5", "luacheck"] | ||
316 | "let g:syntastic_lua_luacheck_args = "--codes --no-max-comment-line-length --ignore 542 551" | ||
317 | " | ||
318 | ""c | ||
319 | ""let g:syntastic_c_compiler =['gcc', 'clang', 'make'] | ||
320 | ""let g:syntastic_c_compiler_options ='-wpedantic -g' | ||
321 | "let g:syntastic_c_compiler_options ='-std=gnu99' | ||
322 | "let g:syntastic_c_include_dirs=['/usr/include/'] | ||
323 | "let g:syntastic_c_config_file='.syntastic_c_config_file' | ||
324 | " | ||
325 | ""cpp | ||
326 | "let g:syntastic_cpp_include_dirs = ['/usr/include/'] | ||
327 | "let g:syntastic_cpp_remove_include_errors = 1 | ||
328 | "let g:syntastic_cpp_check_header = 1 | ||
329 | "let g:syntastic_cpp_compiler = 'clang++' | ||
330 | "let g:syntastic_cpp_compiler_options = '-std=c++11 -stdlib=libstdc++' | ||
331 | "" syntastic end | ||
332 | let g:syntastic_python_checkers=['pylint'] | ||
333 | let g:syntastic_python_pylint_args=['--disable=C0111,R0903,C0301'] | ||
334 | |||
335 | |||
336 | " powerline | ||
337 | let laststatus=2 | ||
338 | let g:airlinr_powerline_fonts=1 "使用官方打过补丁的字体 | ||
339 | let g:Powerline_symbols='fancy' | ||
340 | |||
341 | "vim-youdao-translater | ||
342 | vnoremap <silent> <C-t> :<C-u>Ydv<CR> | ||
343 | nnoremap <silent> <C-t> :<C-u>Ydc<CR> | ||
344 | noremap <leader>yd: <C-u>Yde<CR> | ||
345 | ``` | 205 | ``` |
346 | 206 | ||
347 | ### ./vimrcs/myset.vim | 207 | 我直接把我服务器上的配置文件[放在这里](https://file.qin-juan-ge-zhu.top/useful/vim/),以供查看。 |
348 | |||
349 | ```vim | ||
350 | set nocp "设置兼容 | ||
351 | set expandtab "设置tab | ||
352 | set shiftwidth=4 "设置tab的间隔 | ||
353 | set tabstop=4 "四个空格代表一个tab | ||
354 | set sts=4 | ||
355 | set showmatch "在输入括号时光标会短暂地跳到与之相匹配的括号处 | ||
356 | set autoindent "设置自动缩进 | ||
357 | " set smartindent "设置智能缩进 | ||
358 | set nowrap "设置自动换行 | ||
359 | set tw=500 | ||
360 | set lbr | ||
361 | set foldmethod=indent "设置按缩进折叠代码 | ||
362 | set foldlevel=99 "折叠层级 | ||
363 | nnoremap <space> za "空格代替za进行展开和折叠 | ||
364 | "===set guifont=Monospace "设置字体大小 | ||
365 | set guifont=:b:h16 | ||
366 | set encoding=utf-8 "设置编码为utf-8 | ||
367 | set fileencoding=utf-8 | ||
368 | set fileencodings=ucs-bom,utf-8,GB18030,cp936,big5,euc-jp,euc-kr,latin1 | ||
369 | set helplang=cn "帮助中文支持 | ||
370 | "colorscheme evening "设置主体颜色 | ||
371 | set mouse=a " 设置鼠标 | ||
372 | |||
373 | "自动补全配置 | ||
374 | autocmd FileType python set omnifunc=pythoncomplete#Complete | ||
375 | autocmd FileType javascript set omnifunc=javascriptcomplete#CompleteJS | ||
376 | autocmd FileType html set omnifunc=htmlcomplete#CompleteTags | ||
377 | autocmd FileType css set omnifunc=csscomplete#CompleteCSS | ||
378 | autocmd FileType xml set omnifunc=xmlcomplete#CompleteTags | ||
379 | autocmd FileType php set omnifunc=phpcomplete#CompletePHP | ||
380 | autocmd FileType c set omnifunc=ccomplete#Complete | ||
381 | |||
382 | "进行Taglist的设置 | ||
383 | map <F3> :TlistToggle<CR> | ||
384 | "map <F3> :silent! Tlist<CR> "按下F3就可以呼出了 | ||
385 | "let Tlist_Ctags_Cmd='/usr/bin/ctags' "因为我们放在环境变量里,所以可以直接执行 | ||
386 | let Tlist_Use_Right_Window=1 "让窗口显示在右边,0的话就是显示在左边 | ||
387 | "let Tlist_Show_One_File=1 "让taglist可以同时展示多个文件的函数列表 | ||
388 | "let Tlist_File_Fold_Auto_Close=1 "非当前文件,函数列表折叠隐藏 | ||
389 | "let Tlist_Exit_OnlyWindow=1 "当taglist是最后一个分割窗口时,自动推出vim | ||
390 | "let Tlist_Process_File_Always=0 "是否一直处理tags.1:处理;0:不处理 | ||
391 | "let Tlist_Inc_Winwidth=0 "不是一直实时更新tags,因为没有必要 | ||
392 | |||
393 | set rnu "设置相对行号 | ||
394 | set nu "设置绝对行号 | ||
395 | " Set cursor shape and color | ||
396 | if &term =~ "xterm" | ||
397 | " INSERT mode | ||
398 | let &t_SI = "\<Esc>[6 q" . "\<Esc>]12;blue\x7" | ||
399 | " REPLACE mode | ||
400 | let &t_SR = "\<Esc>[3 q" . "\<Esc>]12;black\x7" | ||
401 | " NORMAL mode | ||
402 | let &t_EI = "\<Esc>[2 q" . "\<Esc>]12;green\x7" | ||
403 | endif | ||
404 | " 1 -> blinking block 闪烁的方块 | ||
405 | " 2 -> solid block 不闪烁的方块 | ||
406 | " 3 -> blinking underscore 闪烁的下划线 | ||
407 | " 4 -> solid underscore 不闪烁的下划线 | ||
408 | " 5 -> blinking vertical bar 闪烁的竖线 | ||
409 | " 6 -> solid vertical bar 不闪烁的竖线 | ||
410 | |||
411 | syntax on | ||
412 | set cul | ||
413 | set cuc | ||
414 | hi CursorLine cterm=NONE ctermbg=darkred ctermfg=white guibg=darkred guifg=white | ||
415 | ""set lines=35 columns=118 | ||
416 | colorscheme zaibatsu "设置颜色主题 | ||
417 | set wrap "设置自动折行 | ||
418 | ``` | ||
419 | |||
420 | ### ./vimrcs/match.vim | ||
421 | |||
422 | ```vim | ||
423 | "各类符号自动匹配 | ||
424 | :inoremap ( ()<ESC>i | ||
425 | :inoremap ) <c-r>=ClosePair(')')<CR> | ||
426 | :inoremap { {}<ESC>i | ||
427 | :inoremap } <c-r>=ClosePair('}')<CR> | ||
428 | :inoremap [ []<ESC>i | ||
429 | :inoremap ] <c-r>=ClosePair(']')<CR> | ||
430 | |||
431 | function ClosePair(char) | ||
432 | if getline('.')[col('.') - 1] == a:char | ||
433 | return "<Right>" | ||
434 | else | ||
435 | return a:char | ||
436 | endif | ||
437 | endfunction | ||
438 | ``` | ||
439 | |||
440 | ### ./vimrcs/fuck.vim | ||
441 | |||
442 | ```vim | ||
443 | " 键盘映射设置区域 | ||
444 | |||
445 | " 对应使用函数的配置 | ||
446 | set guitablabel=%{ShortTabLabel()} | ||
447 | function ShortTabLabel () | ||
448 | let bufnrlist = tabpagebuflist (v:lnum) | ||
449 | let label = bufname (bufnrlist[tabpagewinnr (v:lnum) -1]) | ||
450 | let filename = fnamemodify (label, ':t') | ||
451 | return filename | ||
452 | endfunction | ||
453 | |||
454 | set tabline=%!MyTabLine() | ||
455 | function MyTabLine() | ||
456 | let s = '' | ||
457 | for i in range(tabpagenr('$')) | ||
458 | " 选择高亮 | ||
459 | if i + 1 == tabpagenr() | ||
460 | let s .= '%#TabLineSel#' | ||
461 | else | ||
462 | let s .= '%#TabLine#' | ||
463 | endif | ||
464 | " 设置标签页号 (用于鼠标点击) | ||
465 | let s .= '%' . (i + 1) . 'T' | ||
466 | " MyTabLabel() 提供完整路径标签 MyShortTabLabel 提供文件名标签 | ||
467 | let s .= ' %{MyShortTabLabel(' . (i + 1) . ')} ' | ||
468 | endfor | ||
469 | " 最后一个标签页之后用 TabLineFill 填充并复位标签页号 | ||
470 | let s .= '%#TabLineFill#%T' | ||
471 | " 右对齐用于关闭当前标签页的标签 | ||
472 | if tabpagenr('$') > 1 | ||
473 | let s .= '%=%#TabLine#%999Xclose' | ||
474 | endif | ||
475 | return s | ||
476 | endfunction | ||
477 | " 文件名标签 | ||
478 | function MyShortTabLabel(n) | ||
479 | let buflist = tabpagebuflist(a:n) | ||
480 | let label = bufname (buflist[tabpagewinnr (a:n) -1]) | ||
481 | let filename = fnamemodify (label, ':t') | ||
482 | return filename | ||
483 | endfunction | ||
484 | "完整路径标签 | ||
485 | function MyTabLabel(n) | ||
486 | let buflist = tabpagebuflist(a:n) | ||
487 | let winnr = tabpagewinnr(a:n) | ||
488 | return bufname(buflist[winnr - 1]) | ||
489 | endfunction | ||
490 | " vim 标签样式 | ||
491 | " TabLineFill tab pages line, where there are no labels | ||
492 | hi TabLineFill term=none | ||
493 | hi TabLineFill ctermfg=DarkGrey | ||
494 | hi TabLineFill guifg=#777777 | ||
495 | " TabLineSel tab pages line, active tab page label | ||
496 | hi TabLineSel term=inverse | ||
497 | hi TabLineSel cterm=none ctermfg=yellow ctermbg=Black | ||
498 | hi TabLineSel gui=none guifg=yellow guibg=Black | ||
499 | |||
500 | " Develop editing options | ||
501 | au FileType vim setl expandtab | ||
502 | au FileType vim setl shiftwidth=2 | ||
503 | au FileType vim setl tabstop=2 | ||
504 | |||
505 | " 显示状态栏(默认值为 1,无法显示状态栏) | ||
506 | set laststatus=2 | ||
507 | " Format the statusline | ||
508 | " set statusline= %f %m %r %h %w CWD: %r%{CurDir()}E5%h Line: %l/%L:%c | ||
509 | |||
510 | function! CurDir() | ||
511 | let curdir = substitute(getcwd(), '/Users/amir/', "~/", "g") | ||
512 | return curdir | ||
513 | endfunction | ||
514 | ``` | ||
515 | |||
516 | ### ./vimrcs/keybind.vim | ||
517 | |||
518 | ```vim | ||
519 | map - dd | ||
520 | map up 20k | ||
521 | map ne 20j | ||
522 | map end G | ||
523 | map ta $ | ||
524 | "map <C-l> <C-w-l> | ||
525 | nmap <C-tab> :tabn<CR> | ||
526 | nmap <tab> :tabp<CR> | ||
527 | "nmap <alt-tab> :tabnew<CR> | ||
528 | map <space>= <buffer><esc>:vertical resize +5<cr> | ||
529 | map <space>- <esc>:vertical resize -5<cr> | ||
530 | |||
531 | nn <M-1> 1gt | ||
532 | nn <M-2> 2gt | ||
533 | nn <M-3> 3gt | ||
534 | nn <M-4> 4gt | ||
535 | nn <M-5> 5gt | ||
536 | nn <M-6> 6gt | ||
537 | nn <M-7> 7gt | ||
538 | nn <M-8> 8gt | ||
539 | nn <M-9> 9gt | ||
540 | nn <M-0> :tablast<CR> | ||
541 | |||
542 | "光标在页面间切换 | ||
543 | nnoremap <space>h <C-w><C-h> | ||
544 | nnoremap <space>j <C-w><C-j> | ||
545 | nnoremap <space>k <C-w><C-k> | ||
546 | nnoremap <space>l <C-w><C-l> | ||
547 | |||
548 | "复制粘贴 | ||
549 | "vnoremap <C-y> "+y | ||
550 | "nnoremap <C-p> "+p | ||
551 | |||
552 | " 其他开发配置 | ||
553 | " Python | ||
554 | au BufNewFile,BufRead *.py | ||
555 | \ set tabstop=4 | | ||
556 | \ set shiftwidth=4 | | ||
557 | \ set textwidth=79 | | ||
558 | \ set expandtab | | ||
559 | \ set autoindent | | ||
560 | \ set fileformat=unix | | ||
561 | ``` | ||
562 | |||
563 | ### ./vimrcs/autocmd.vim | ||
564 | |||
565 | ```vim | ||
566 | autocmd BufReadPost,BufWritePre *.html,*.c,*.cpp normal gg=G | ||
567 | autocmd Filetype html,markdown setlocal nowrap | ||
568 | "autocmd Filetype c,cpp map <buffer><C-p> I//<esc> | ||
569 | autocmd Filetype c,cpp map <buffer><C-d> 0xx<esc>ta | ||
570 | "autocmd Filetype c,cpp vmap <buffer><C-p> <C-v>I//<esc> | ||
571 | "autocmd Filetype python map <buffer><C-p> I# <esc> | ||
572 | autocmd Filetype python map <buffer><C-d> 0xx<esc>ta | ||
573 | "autocmd Filetype python vmap <buffer><C-p> <C-v>I# <esc> | ||
574 | "autocmd Filetype c,cpp vmap <buffer><C-d> <C-v>0xx | ||
575 | "说明:本文件上文所有<Ctrl-p>的设置,原是用来为c,cpp,python文件设置注释的 | ||
576 | "但是注释可以通过<Ctrl-v>进入块选模式而后<Ctrl-i>进入插入模式,在前边输入注释符号,保存,则选中的行均被注释 | ||
577 | "甚为方便,因此这许多键一起弃用。 | ||
578 | ``` | ||
diff --git a/code/stdc++.html b/code/stdc++.html deleted file mode 100644 index 1706db0..0000000 --- a/code/stdc++.html +++ /dev/null | |||
@@ -1,892 +0,0 @@ | |||
1 | <!doctype html> | ||
2 | <html> | ||
3 | |||
4 | <head> | ||
5 | <meta charset='UTF-8'> | ||
6 | <meta name='viewport' content='width=device-width initial-scale=1'> | ||
7 | <title>C++九千能头文件</title> | ||
8 | <link rel="stylesheet" href="https://www.qin-juan-ge-zhu.top/common/CSS/github.css"> | ||
9 | <script type="text/javascript" src="https://www.qin-juan-ge-zhu.top/common/script4code.js"></script> | ||
10 | </head> | ||
11 | |||
12 | <body class='typora-export os-windows'> | ||
13 | <div class='typora-export-content'> | ||
14 | <div id='write' class=''> | ||
15 | <pre class="md-fences md-end-block md-fences-with-lineno ty-contain-cm modeLoaded" spellcheck="false" | ||
16 | lang="cpp" | ||
17 | style="break-inside: unset;"><div class="CodeMirror cm-s-inner cm-s-null-scroll CodeMirror-wrap" lang="cpp"><div style="overflow: hidden; position: relative; width: 3px; height: 0px; top: 9.51875px; left: 53px;"><textarea autocorrect="off" autocapitalize="off" spellcheck="false" tabindex="0" style="position: absolute; bottom: -1em; padding: 0px; width: 1000px; height: 1em; outline: none;"></textarea></div><div class="CodeMirror-scrollbar-filler" cm-not-content="true"></div><div class="CodeMirror-gutter-filler" cm-not-content="true"></div><div class="CodeMirror-scroll" tabindex="-1"><div class="CodeMirror-sizer" style="margin-left: 45px; margin-bottom: 0px; border-right-width: 0px; padding-right: 0px; padding-bottom: 0px;"><div style="position: relative; top: 0px;"><div class="CodeMirror-lines" role="presentation"><div role="presentation" style="position: relative; outline: none;"><div class="CodeMirror-measure"><pre><span>xxxxxxxxxx</span></pre> | ||
18 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt"> | ||
19 | <div>120</div> | ||
20 | </div> | ||
21 | </div> | ||
22 | <div class="CodeMirror-measure"></div> | ||
23 | <div style="position: relative; z-index: 1;"></div> | ||
24 | <div class="CodeMirror-code" role="presentation" style=""> | ||
25 | <div class="CodeMirror-activeline" style="position: relative;"> | ||
26 | <div class="CodeMirror-activeline-background CodeMirror-linebackground"></div> | ||
27 | <div class="CodeMirror-gutter-background CodeMirror-activeline-gutter" | ||
28 | style="left: -45px; width: 45px;"></div> | ||
29 | <div class="CodeMirror-gutter-wrapper CodeMirror-activeline-gutter" style="left: -45px;"> | ||
30 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt CodeMirror-linenumber-show" | ||
31 | style="left: 0px; width: 36px;">1</div> | ||
32 | </div> | ||
33 | <pre class=" CodeMirror-line " | ||
34 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-comment">// C++ includes used for precompiling -*- C++ -*-</span></span></pre> | ||
35 | </div> | ||
36 | <div style="position: relative;"> | ||
37 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
38 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">2</div> | ||
39 | </div> | ||
40 | <pre class=" CodeMirror-line " | ||
41 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"> </span></pre> | ||
42 | </div> | ||
43 | <div style="position: relative;"> | ||
44 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
45 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">3</div> | ||
46 | </div> | ||
47 | <pre class=" CodeMirror-line " | ||
48 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-comment">// Copyright (C) 2003-2013 Free Software Foundation, Inc.</span></span></pre> | ||
49 | </div> | ||
50 | <div style="position: relative;"> | ||
51 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
52 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">4</div> | ||
53 | </div> | ||
54 | <pre class=" CodeMirror-line " | ||
55 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-comment">//</span></span></pre> | ||
56 | </div> | ||
57 | <div style="position: relative;"> | ||
58 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
59 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">5</div> | ||
60 | </div> | ||
61 | <pre class=" CodeMirror-line " | ||
62 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-comment">// This file is part of the GNU ISO C++ Library. This library is free</span></span></pre> | ||
63 | </div> | ||
64 | <div style="position: relative;"> | ||
65 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
66 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">6</div> | ||
67 | </div> | ||
68 | <pre class=" CodeMirror-line " | ||
69 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-comment">// software; you can redistribute it and/or modify it under the</span></span></pre> | ||
70 | </div> | ||
71 | <div style="position: relative;"> | ||
72 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
73 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">7</div> | ||
74 | </div> | ||
75 | <pre class=" CodeMirror-line " | ||
76 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-comment">// terms of the GNU General Public License as published by the</span></span></pre> | ||
77 | </div> | ||
78 | <div style="position: relative;"> | ||
79 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
80 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">8</div> | ||
81 | </div> | ||
82 | <pre class=" CodeMirror-line " | ||
83 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-comment">// Free Software Foundation; either version 3, or (at your option)</span></span></pre> | ||
84 | </div> | ||
85 | <div style="position: relative;"> | ||
86 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
87 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">9</div> | ||
88 | </div> | ||
89 | <pre class=" CodeMirror-line " | ||
90 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-comment">// any later version.</span></span></pre> | ||
91 | </div> | ||
92 | <div style="position: relative;"> | ||
93 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
94 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt CodeMirror-linenumber-show" | ||
95 | style="left: 0px; width: 36px;">10</div> | ||
96 | </div> | ||
97 | <pre class=" CodeMirror-line " | ||
98 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"> </span></pre> | ||
99 | </div> | ||
100 | <div style="position: relative;"> | ||
101 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
102 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">11</div> | ||
103 | </div> | ||
104 | <pre class=" CodeMirror-line " | ||
105 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-comment">// This library is distributed in the hope that it will be useful,</span></span></pre> | ||
106 | </div> | ||
107 | <div style="position: relative;"> | ||
108 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
109 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">12</div> | ||
110 | </div> | ||
111 | <pre class=" CodeMirror-line " | ||
112 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-comment">// but WITHOUT ANY WARRANTY; without even the implied warranty of</span></span></pre> | ||
113 | </div> | ||
114 | <div style="position: relative;"> | ||
115 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
116 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">13</div> | ||
117 | </div> | ||
118 | <pre class=" CodeMirror-line " | ||
119 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-comment">// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the</span></span></pre> | ||
120 | </div> | ||
121 | <div style="position: relative;"> | ||
122 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
123 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">14</div> | ||
124 | </div> | ||
125 | <pre class=" CodeMirror-line " | ||
126 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-comment">// GNU General Public License for more details.</span></span></pre> | ||
127 | </div> | ||
128 | <div style="position: relative;"> | ||
129 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
130 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">15</div> | ||
131 | </div> | ||
132 | <pre class=" CodeMirror-line " | ||
133 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"> </span></pre> | ||
134 | </div> | ||
135 | <div style="position: relative;"> | ||
136 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
137 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">16</div> | ||
138 | </div> | ||
139 | <pre class=" CodeMirror-line " | ||
140 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-comment">// Under Section 7 of GPL version 3, you are granted additional</span></span></pre> | ||
141 | </div> | ||
142 | <div style="position: relative;"> | ||
143 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
144 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">17</div> | ||
145 | </div> | ||
146 | <pre class=" CodeMirror-line " | ||
147 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-comment">// permissions described in the GCC Runtime Library Exception, version</span></span></pre> | ||
148 | </div> | ||
149 | <div style="position: relative;"> | ||
150 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
151 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">18</div> | ||
152 | </div> | ||
153 | <pre class=" CodeMirror-line " | ||
154 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-comment">// 3.1, as published by the Free Software Foundation.</span></span></pre> | ||
155 | </div> | ||
156 | <div style="position: relative;"> | ||
157 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
158 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">19</div> | ||
159 | </div> | ||
160 | <pre class=" CodeMirror-line " | ||
161 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"> </span></pre> | ||
162 | </div> | ||
163 | <div style="position: relative;"> | ||
164 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
165 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt CodeMirror-linenumber-show" | ||
166 | style="left: 0px; width: 36px;">20</div> | ||
167 | </div> | ||
168 | <pre class=" CodeMirror-line " | ||
169 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-comment">// You should have received a copy of the GNU General Public License and</span></span></pre> | ||
170 | </div> | ||
171 | <div style="position: relative;"> | ||
172 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
173 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">21</div> | ||
174 | </div> | ||
175 | <pre class=" CodeMirror-line " | ||
176 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-comment">// a copy of the GCC Runtime Library Exception along with this program;</span></span></pre> | ||
177 | </div> | ||
178 | <div style="position: relative;"> | ||
179 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
180 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">22</div> | ||
181 | </div> | ||
182 | <pre class=" CodeMirror-line " | ||
183 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-comment">// see the files COPYING3 and COPYING.RUNTIME respectively. If not, see</span></span></pre> | ||
184 | </div> | ||
185 | <div style="position: relative;"> | ||
186 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
187 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">23</div> | ||
188 | </div> | ||
189 | <pre class=" CodeMirror-line " | ||
190 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-comment">// <Licenses - GNU Project - Free Software Foundation>.</span></span></pre> | ||
191 | </div> | ||
192 | <div style="position: relative;"> | ||
193 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
194 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">24</div> | ||
195 | </div> | ||
196 | <pre class=" CodeMirror-line " | ||
197 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"> </span></pre> | ||
198 | </div> | ||
199 | <div style="position: relative;"> | ||
200 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
201 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">25</div> | ||
202 | </div> | ||
203 | <pre class=" CodeMirror-line " | ||
204 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-comment">/** @file stdc++.h</span></span></pre> | ||
205 | </div> | ||
206 | <div style="position: relative;"> | ||
207 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
208 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">26</div> | ||
209 | </div> | ||
210 | <pre class=" CodeMirror-line " | ||
211 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"> <span class="cm-comment">* This is an implementation file for a precompiled header.</span></span></pre> | ||
212 | </div> | ||
213 | <div style="position: relative;"> | ||
214 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
215 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">27</div> | ||
216 | </div> | ||
217 | <pre class=" CodeMirror-line " | ||
218 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"> <span class="cm-comment">*/</span></span></pre> | ||
219 | </div> | ||
220 | <div style="position: relative;"> | ||
221 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
222 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">28</div> | ||
223 | </div> | ||
224 | <pre class=" CodeMirror-line " | ||
225 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"> </span></pre> | ||
226 | </div> | ||
227 | <div style="position: relative;"> | ||
228 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
229 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">29</div> | ||
230 | </div> | ||
231 | <pre class=" CodeMirror-line " | ||
232 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-comment">// 17.4.1.2 Headers</span></span></pre> | ||
233 | </div> | ||
234 | <div style="position: relative;"> | ||
235 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
236 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt CodeMirror-linenumber-show" | ||
237 | style="left: 0px; width: 36px;">30</div> | ||
238 | </div> | ||
239 | <pre class=" CodeMirror-line " | ||
240 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"> </span></pre> | ||
241 | </div> | ||
242 | <div style="position: relative;"> | ||
243 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
244 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">31</div> | ||
245 | </div> | ||
246 | <pre class=" CodeMirror-line " | ||
247 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-comment">// C</span></span></pre> | ||
248 | </div> | ||
249 | <div style="position: relative;"> | ||
250 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
251 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">32</div> | ||
252 | </div> | ||
253 | <pre class=" CodeMirror-line " | ||
254 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#ifndef _GLIBCXX_NO_ASSERT</span></span></pre> | ||
255 | </div> | ||
256 | <div style="position: relative;"> | ||
257 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
258 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">33</div> | ||
259 | </div> | ||
260 | <pre class=" CodeMirror-line " | ||
261 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <cassert></span></span></pre> | ||
262 | </div> | ||
263 | <div style="position: relative;"> | ||
264 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
265 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">34</div> | ||
266 | </div> | ||
267 | <pre class=" CodeMirror-line " | ||
268 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#endif</span></span></pre> | ||
269 | </div> | ||
270 | <div style="position: relative;"> | ||
271 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
272 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">35</div> | ||
273 | </div> | ||
274 | <pre class=" CodeMirror-line " | ||
275 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <cctype></span></span></pre> | ||
276 | </div> | ||
277 | <div style="position: relative;"> | ||
278 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
279 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">36</div> | ||
280 | </div> | ||
281 | <pre class=" CodeMirror-line " | ||
282 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <cerrno></span></span></pre> | ||
283 | </div> | ||
284 | <div style="position: relative;"> | ||
285 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
286 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">37</div> | ||
287 | </div> | ||
288 | <pre class=" CodeMirror-line " | ||
289 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <cfloat></span></span></pre> | ||
290 | </div> | ||
291 | <div style="position: relative;"> | ||
292 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
293 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">38</div> | ||
294 | </div> | ||
295 | <pre class=" CodeMirror-line " | ||
296 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <ciso646></span></span></pre> | ||
297 | </div> | ||
298 | <div style="position: relative;"> | ||
299 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
300 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">39</div> | ||
301 | </div> | ||
302 | <pre class=" CodeMirror-line " | ||
303 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <climits></span></span></pre> | ||
304 | </div> | ||
305 | <div style="position: relative;"> | ||
306 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
307 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt CodeMirror-linenumber-show" | ||
308 | style="left: 0px; width: 36px;">40</div> | ||
309 | </div> | ||
310 | <pre class=" CodeMirror-line " | ||
311 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <clocale></span></span></pre> | ||
312 | </div> | ||
313 | <div style="position: relative;"> | ||
314 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
315 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">41</div> | ||
316 | </div> | ||
317 | <pre class=" CodeMirror-line " | ||
318 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <cmath></span></span></pre> | ||
319 | </div> | ||
320 | <div style="position: relative;"> | ||
321 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
322 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">42</div> | ||
323 | </div> | ||
324 | <pre class=" CodeMirror-line " | ||
325 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <csetjmp></span></span></pre> | ||
326 | </div> | ||
327 | <div style="position: relative;"> | ||
328 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
329 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">43</div> | ||
330 | </div> | ||
331 | <pre class=" CodeMirror-line " | ||
332 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <csignal></span></span></pre> | ||
333 | </div> | ||
334 | <div style="position: relative;"> | ||
335 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
336 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">44</div> | ||
337 | </div> | ||
338 | <pre class=" CodeMirror-line " | ||
339 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <cstdarg></span></span></pre> | ||
340 | </div> | ||
341 | <div style="position: relative;"> | ||
342 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
343 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">45</div> | ||
344 | </div> | ||
345 | <pre class=" CodeMirror-line " | ||
346 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <cstddef></span></span></pre> | ||
347 | </div> | ||
348 | <div style="position: relative;"> | ||
349 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
350 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">46</div> | ||
351 | </div> | ||
352 | <pre class=" CodeMirror-line " | ||
353 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <cstdio></span></span></pre> | ||
354 | </div> | ||
355 | <div style="position: relative;"> | ||
356 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
357 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">47</div> | ||
358 | </div> | ||
359 | <pre class=" CodeMirror-line " | ||
360 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <cstdlib></span></span></pre> | ||
361 | </div> | ||
362 | <div style="position: relative;"> | ||
363 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
364 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">48</div> | ||
365 | </div> | ||
366 | <pre class=" CodeMirror-line " | ||
367 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <cstring></span></span></pre> | ||
368 | </div> | ||
369 | <div style="position: relative;"> | ||
370 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
371 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">49</div> | ||
372 | </div> | ||
373 | <pre class=" CodeMirror-line " | ||
374 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <ctime></span></span></pre> | ||
375 | </div> | ||
376 | <div style="position: relative;"> | ||
377 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
378 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt CodeMirror-linenumber-show" | ||
379 | style="left: 0px; width: 36px;">50</div> | ||
380 | </div> | ||
381 | <pre class=" CodeMirror-line " | ||
382 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"> </span></pre> | ||
383 | </div> | ||
384 | <div style="position: relative;"> | ||
385 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
386 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">51</div> | ||
387 | </div> | ||
388 | <pre class=" CodeMirror-line " | ||
389 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#if __cplusplus >= 201103L</span></span></pre> | ||
390 | </div> | ||
391 | <div style="position: relative;"> | ||
392 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
393 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">52</div> | ||
394 | </div> | ||
395 | <pre class=" CodeMirror-line " | ||
396 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <ccomplex></span></span></pre> | ||
397 | </div> | ||
398 | <div style="position: relative;"> | ||
399 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
400 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">53</div> | ||
401 | </div> | ||
402 | <pre class=" CodeMirror-line " | ||
403 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <cfenv></span></span></pre> | ||
404 | </div> | ||
405 | <div style="position: relative;"> | ||
406 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
407 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">54</div> | ||
408 | </div> | ||
409 | <pre class=" CodeMirror-line " | ||
410 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <cinttypes></span></span></pre> | ||
411 | </div> | ||
412 | <div style="position: relative;"> | ||
413 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
414 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">55</div> | ||
415 | </div> | ||
416 | <pre class=" CodeMirror-line " | ||
417 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <cstdalign></span></span></pre> | ||
418 | </div> | ||
419 | <div style="position: relative;"> | ||
420 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
421 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">56</div> | ||
422 | </div> | ||
423 | <pre class=" CodeMirror-line " | ||
424 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <cstdbool></span></span></pre> | ||
425 | </div> | ||
426 | <div style="position: relative;"> | ||
427 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
428 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">57</div> | ||
429 | </div> | ||
430 | <pre class=" CodeMirror-line " | ||
431 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <cstdint></span></span></pre> | ||
432 | </div> | ||
433 | <div style="position: relative;"> | ||
434 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
435 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">58</div> | ||
436 | </div> | ||
437 | <pre class=" CodeMirror-line " | ||
438 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <ctgmath></span></span></pre> | ||
439 | </div> | ||
440 | <div style="position: relative;"> | ||
441 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
442 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">59</div> | ||
443 | </div> | ||
444 | <pre class=" CodeMirror-line " | ||
445 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <cwchar></span></span></pre> | ||
446 | </div> | ||
447 | <div style="position: relative;"> | ||
448 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
449 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt CodeMirror-linenumber-show" | ||
450 | style="left: 0px; width: 36px;">60</div> | ||
451 | </div> | ||
452 | <pre class=" CodeMirror-line " | ||
453 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <cwctype></span></span></pre> | ||
454 | </div> | ||
455 | <div style="position: relative;"> | ||
456 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
457 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">61</div> | ||
458 | </div> | ||
459 | <pre class=" CodeMirror-line " | ||
460 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#endif</span></span></pre> | ||
461 | </div> | ||
462 | <div style="position: relative;"> | ||
463 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
464 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">62</div> | ||
465 | </div> | ||
466 | <pre class=" CodeMirror-line " | ||
467 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"> </span></pre> | ||
468 | </div> | ||
469 | <div style="position: relative;"> | ||
470 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
471 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">63</div> | ||
472 | </div> | ||
473 | <pre class=" CodeMirror-line " | ||
474 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-comment">// C++</span></span></pre> | ||
475 | </div> | ||
476 | <div style="position: relative;"> | ||
477 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
478 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">64</div> | ||
479 | </div> | ||
480 | <pre class=" CodeMirror-line " | ||
481 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <algorithm></span></span></pre> | ||
482 | </div> | ||
483 | <div style="position: relative;"> | ||
484 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
485 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">65</div> | ||
486 | </div> | ||
487 | <pre class=" CodeMirror-line " | ||
488 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <bitset></span></span></pre> | ||
489 | </div> | ||
490 | <div style="position: relative;"> | ||
491 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
492 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">66</div> | ||
493 | </div> | ||
494 | <pre class=" CodeMirror-line " | ||
495 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <complex></span></span></pre> | ||
496 | </div> | ||
497 | <div style="position: relative;"> | ||
498 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
499 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">67</div> | ||
500 | </div> | ||
501 | <pre class=" CodeMirror-line " | ||
502 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <deque></span></span></pre> | ||
503 | </div> | ||
504 | <div style="position: relative;"> | ||
505 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
506 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">68</div> | ||
507 | </div> | ||
508 | <pre class=" CodeMirror-line " | ||
509 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <exception></span></span></pre> | ||
510 | </div> | ||
511 | <div style="position: relative;"> | ||
512 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
513 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">69</div> | ||
514 | </div> | ||
515 | <pre class=" CodeMirror-line " | ||
516 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <fstream></span></span></pre> | ||
517 | </div> | ||
518 | <div style="position: relative;"> | ||
519 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
520 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt CodeMirror-linenumber-show" | ||
521 | style="left: 0px; width: 36px;">70</div> | ||
522 | </div> | ||
523 | <pre class=" CodeMirror-line " | ||
524 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <functional></span></span></pre> | ||
525 | </div> | ||
526 | <div style="position: relative;"> | ||
527 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
528 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">71</div> | ||
529 | </div> | ||
530 | <pre class=" CodeMirror-line " | ||
531 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <iomanip></span></span></pre> | ||
532 | </div> | ||
533 | <div style="position: relative;"> | ||
534 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
535 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">72</div> | ||
536 | </div> | ||
537 | <pre class=" CodeMirror-line " | ||
538 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <ios></span></span></pre> | ||
539 | </div> | ||
540 | <div style="position: relative;"> | ||
541 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
542 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">73</div> | ||
543 | </div> | ||
544 | <pre class=" CodeMirror-line " | ||
545 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <iosfwd></span></span></pre> | ||
546 | </div> | ||
547 | <div style="position: relative;"> | ||
548 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
549 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">74</div> | ||
550 | </div> | ||
551 | <pre class=" CodeMirror-line " | ||
552 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <iostream></span></span></pre> | ||
553 | </div> | ||
554 | <div style="position: relative;"> | ||
555 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
556 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">75</div> | ||
557 | </div> | ||
558 | <pre class=" CodeMirror-line " | ||
559 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <istream></span></span></pre> | ||
560 | </div> | ||
561 | <div style="position: relative;"> | ||
562 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
563 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">76</div> | ||
564 | </div> | ||
565 | <pre class=" CodeMirror-line " | ||
566 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <iterator></span></span></pre> | ||
567 | </div> | ||
568 | <div style="position: relative;"> | ||
569 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
570 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">77</div> | ||
571 | </div> | ||
572 | <pre class=" CodeMirror-line " | ||
573 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <limits></span></span></pre> | ||
574 | </div> | ||
575 | <div style="position: relative;"> | ||
576 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
577 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">78</div> | ||
578 | </div> | ||
579 | <pre class=" CodeMirror-line " | ||
580 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <list></span></span></pre> | ||
581 | </div> | ||
582 | <div style="position: relative;"> | ||
583 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
584 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">79</div> | ||
585 | </div> | ||
586 | <pre class=" CodeMirror-line " | ||
587 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <locale></span></span></pre> | ||
588 | </div> | ||
589 | <div style="position: relative;"> | ||
590 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
591 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt CodeMirror-linenumber-show" | ||
592 | style="left: 0px; width: 36px;">80</div> | ||
593 | </div> | ||
594 | <pre class=" CodeMirror-line " | ||
595 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <map></span></span></pre> | ||
596 | </div> | ||
597 | <div style="position: relative;"> | ||
598 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
599 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">81</div> | ||
600 | </div> | ||
601 | <pre class=" CodeMirror-line " | ||
602 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <memory></span></span></pre> | ||
603 | </div> | ||
604 | <div style="position: relative;"> | ||
605 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
606 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">82</div> | ||
607 | </div> | ||
608 | <pre class=" CodeMirror-line " | ||
609 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <new></span></span></pre> | ||
610 | </div> | ||
611 | <div style="position: relative;"> | ||
612 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
613 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">83</div> | ||
614 | </div> | ||
615 | <pre class=" CodeMirror-line " | ||
616 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <numeric></span></span></pre> | ||
617 | </div> | ||
618 | <div style="position: relative;"> | ||
619 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
620 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">84</div> | ||
621 | </div> | ||
622 | <pre class=" CodeMirror-line " | ||
623 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <ostream></span></span></pre> | ||
624 | </div> | ||
625 | <div style="position: relative;"> | ||
626 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
627 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">85</div> | ||
628 | </div> | ||
629 | <pre class=" CodeMirror-line " | ||
630 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <queue></span></span></pre> | ||
631 | </div> | ||
632 | <div style="position: relative;"> | ||
633 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
634 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">86</div> | ||
635 | </div> | ||
636 | <pre class=" CodeMirror-line " | ||
637 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <set></span></span></pre> | ||
638 | </div> | ||
639 | <div style="position: relative;"> | ||
640 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
641 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">87</div> | ||
642 | </div> | ||
643 | <pre class=" CodeMirror-line " | ||
644 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <sstream></span></span></pre> | ||
645 | </div> | ||
646 | <div style="position: relative;"> | ||
647 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
648 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">88</div> | ||
649 | </div> | ||
650 | <pre class=" CodeMirror-line " | ||
651 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <stack></span></span></pre> | ||
652 | </div> | ||
653 | <div style="position: relative;"> | ||
654 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
655 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">89</div> | ||
656 | </div> | ||
657 | <pre class=" CodeMirror-line " | ||
658 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <stdexcept></span></span></pre> | ||
659 | </div> | ||
660 | <div style="position: relative;"> | ||
661 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
662 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt CodeMirror-linenumber-show" | ||
663 | style="left: 0px; width: 36px;">90</div> | ||
664 | </div> | ||
665 | <pre class=" CodeMirror-line " | ||
666 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <streambuf></span></span></pre> | ||
667 | </div> | ||
668 | <div style="position: relative;"> | ||
669 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
670 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">91</div> | ||
671 | </div> | ||
672 | <pre class=" CodeMirror-line " | ||
673 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <string></span></span></pre> | ||
674 | </div> | ||
675 | <div style="position: relative;"> | ||
676 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
677 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">92</div> | ||
678 | </div> | ||
679 | <pre class=" CodeMirror-line " | ||
680 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <thread></span></span></pre> | ||
681 | </div> | ||
682 | <div style="position: relative;"> | ||
683 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
684 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">93</div> | ||
685 | </div> | ||
686 | <pre class=" CodeMirror-line " | ||
687 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <typeinfo></span></span></pre> | ||
688 | </div> | ||
689 | <div style="position: relative;"> | ||
690 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
691 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">94</div> | ||
692 | </div> | ||
693 | <pre class=" CodeMirror-line " | ||
694 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <unistd.h></span></span></pre> | ||
695 | </div> | ||
696 | <div style="position: relative;"> | ||
697 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
698 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">95</div> | ||
699 | </div> | ||
700 | <pre class=" CodeMirror-line " | ||
701 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <utility></span></span></pre> | ||
702 | </div> | ||
703 | <div style="position: relative;"> | ||
704 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
705 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">96</div> | ||
706 | </div> | ||
707 | <pre class=" CodeMirror-line " | ||
708 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <valarray></span></span></pre> | ||
709 | </div> | ||
710 | <div style="position: relative;"> | ||
711 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
712 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">97</div> | ||
713 | </div> | ||
714 | <pre class=" CodeMirror-line " | ||
715 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <vector></span></span></pre> | ||
716 | </div> | ||
717 | <div style="position: relative;"> | ||
718 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
719 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">98</div> | ||
720 | </div> | ||
721 | <pre class=" CodeMirror-line " | ||
722 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"> </span></pre> | ||
723 | </div> | ||
724 | <div style="position: relative;"> | ||
725 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
726 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">99</div> | ||
727 | </div> | ||
728 | <pre class=" CodeMirror-line " | ||
729 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#if __cplusplus >= 201103L</span></span></pre> | ||
730 | </div> | ||
731 | <div style="position: relative;"> | ||
732 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
733 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt CodeMirror-linenumber-show" | ||
734 | style="left: 0px; width: 36px;">100</div> | ||
735 | </div> | ||
736 | <pre class=" CodeMirror-line " | ||
737 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <array></span></span></pre> | ||
738 | </div> | ||
739 | <div style="position: relative;"> | ||
740 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
741 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">101</div> | ||
742 | </div> | ||
743 | <pre class=" CodeMirror-line " | ||
744 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <atomic></span></span></pre> | ||
745 | </div> | ||
746 | <div style="position: relative;"> | ||
747 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
748 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">102</div> | ||
749 | </div> | ||
750 | <pre class=" CodeMirror-line " | ||
751 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <chrono></span></span></pre> | ||
752 | </div> | ||
753 | <div style="position: relative;"> | ||
754 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
755 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">103</div> | ||
756 | </div> | ||
757 | <pre class=" CodeMirror-line " | ||
758 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <condition_variable></span></span></pre> | ||
759 | </div> | ||
760 | <div style="position: relative;"> | ||
761 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
762 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">104</div> | ||
763 | </div> | ||
764 | <pre class=" CodeMirror-line " | ||
765 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <forward_list></span></span></pre> | ||
766 | </div> | ||
767 | <div style="position: relative;"> | ||
768 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
769 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">105</div> | ||
770 | </div> | ||
771 | <pre class=" CodeMirror-line " | ||
772 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <future></span></span></pre> | ||
773 | </div> | ||
774 | <div style="position: relative;"> | ||
775 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
776 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">106</div> | ||
777 | </div> | ||
778 | <pre class=" CodeMirror-line " | ||
779 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <initializer_list></span></span></pre> | ||
780 | </div> | ||
781 | <div style="position: relative;"> | ||
782 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
783 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">107</div> | ||
784 | </div> | ||
785 | <pre class=" CodeMirror-line " | ||
786 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <mutex></span></span></pre> | ||
787 | </div> | ||
788 | <div style="position: relative;"> | ||
789 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
790 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">108</div> | ||
791 | </div> | ||
792 | <pre class=" CodeMirror-line " | ||
793 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <random></span></span></pre> | ||
794 | </div> | ||
795 | <div style="position: relative;"> | ||
796 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
797 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">109</div> | ||
798 | </div> | ||
799 | <pre class=" CodeMirror-line " | ||
800 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <ratio></span></span></pre> | ||
801 | </div> | ||
802 | <div style="position: relative;"> | ||
803 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
804 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt CodeMirror-linenumber-show" | ||
805 | style="left: 0px; width: 36px;">110</div> | ||
806 | </div> | ||
807 | <pre class=" CodeMirror-line " | ||
808 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <regex></span></span></pre> | ||
809 | </div> | ||
810 | <div style="position: relative;"> | ||
811 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
812 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">111</div> | ||
813 | </div> | ||
814 | <pre class=" CodeMirror-line " | ||
815 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <scoped_allocator></span></span></pre> | ||
816 | </div> | ||
817 | <div style="position: relative;"> | ||
818 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
819 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">112</div> | ||
820 | </div> | ||
821 | <pre class=" CodeMirror-line " | ||
822 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <system_error></span></span></pre> | ||
823 | </div> | ||
824 | <div style="position: relative;"> | ||
825 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
826 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">113</div> | ||
827 | </div> | ||
828 | <pre class=" CodeMirror-line " | ||
829 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <thread></span></span></pre> | ||
830 | </div> | ||
831 | <div style="position: relative;"> | ||
832 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
833 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">114</div> | ||
834 | </div> | ||
835 | <pre class=" CodeMirror-line " | ||
836 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <tuple></span></span></pre> | ||
837 | </div> | ||
838 | <div style="position: relative;"> | ||
839 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
840 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">115</div> | ||
841 | </div> | ||
842 | <pre class=" CodeMirror-line " | ||
843 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <typeindex></span></span></pre> | ||
844 | </div> | ||
845 | <div style="position: relative;"> | ||
846 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
847 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">116</div> | ||
848 | </div> | ||
849 | <pre class=" CodeMirror-line " | ||
850 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <type_traits></span></span></pre> | ||
851 | </div> | ||
852 | <div style="position: relative;"> | ||
853 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
854 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">117</div> | ||
855 | </div> | ||
856 | <pre class=" CodeMirror-line " | ||
857 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <unordered_map></span></span></pre> | ||
858 | </div> | ||
859 | <div style="position: relative;"> | ||
860 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
861 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt" style="left: 0px; width: 36px;">118</div> | ||
862 | </div> | ||
863 | <pre class=" CodeMirror-line " | ||
864 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#include <unordered_set></span></span></pre> | ||
865 | </div> | ||
866 | <div class="" style="position: relative;"> | ||
867 | <div class="CodeMirror-gutter-wrapper" style="left: -45px;"> | ||
868 | <div class="CodeMirror-linenumber CodeMirror-gutter-elt CodeMirror-linenumber-show" | ||
869 | style="left: 0px; width: 36px;">119</div> | ||
870 | </div> | ||
871 | <pre class=" CodeMirror-line " | ||
872 | role="presentation"><span role="presentation" style="padding-right: 0.1px;"><span class="cm-meta">#endif</span></span></pre> | ||
873 | </div> | ||
874 | </div> | ||
875 | </div> | ||
876 | </div> | ||
877 | </div> | ||
878 | </div> | ||
879 | <div style="position: absolute; height: 0px; width: 1px; border-bottom: 0px solid transparent; top: 2741px;"></div> | ||
880 | <div class="CodeMirror-gutters" style="height: 2741px;"> | ||
881 | <div class="CodeMirror-gutter CodeMirror-linenumbers" style="width: 44px;"></div> | ||
882 | </div> | ||
883 | </div> | ||
884 | </div> | ||
885 | </pre> | ||
886 | <p> </p> | ||
887 | <script src="https://www.qin-juan-ge-zhu.top/common/js/comment.js"></script> | ||
888 | </div> | ||
889 | </div> | ||
890 | </body> | ||
891 | |||
892 | </html> \ No newline at end of file | ||
diff --git a/common/CSS/MySet.css b/common/CSS/MySet.css index 19068b1..e152141 100644 --- a/common/CSS/MySet.css +++ b/common/CSS/MySet.css | |||
@@ -1,4 +1,7 @@ | |||
1 | /* 以下部分为自行开发 */ | 1 | /* 以下部分为自行开发 */ |
2 | .wall{ | ||
3 | width:100%; | ||
4 | } | ||
2 | 5 | ||
3 | a { | 6 | a { |
4 | text-decoration: none !important; | 7 | text-decoration: none !important; |
diff --git a/common/CSS/huang.css b/common/CSS/huang.css index 1142414..393d818 100644 --- a/common/CSS/huang.css +++ b/common/CSS/huang.css | |||
@@ -118,6 +118,8 @@ h3 { | |||
118 | 118 | ||
119 | .background { | 119 | .background { |
120 | background: linear-gradient(#4a96cd, #7cb6dd, #4a96cd); | 120 | background: linear-gradient(#4a96cd, #7cb6dd, #4a96cd); |
121 | /* 设置为彩虹色 */ | ||
122 | /* background: linear-gradient(45deg, #ff0000, #ff7f00, #ffff00, #00ff00, #0000ff, #4b0082, #8f00ff); */ | ||
121 | width: 100%; | 123 | width: 100%; |
122 | padding: 0; | 124 | padding: 0; |
123 | margin: 0; | 125 | margin: 0; |
diff --git a/common/script4works.html b/common/script4works.html index fb52f9b..7e74c30 100644 --- a/common/script4works.html +++ b/common/script4works.html | |||
@@ -114,7 +114,6 @@ | |||
114 | 114 | ||
115 | <div class="child" id="child1" style="display:none"> | 115 | <div class="child" id="child1" style="display:none"> |
116 | <a href="https://www.qin-juan-ge-zhu.top/works/index.html">-勤倦阁记</a> | 116 | <a href="https://www.qin-juan-ge-zhu.top/works/index.html">-勤倦阁记</a> |
117 | <a href="https://www.qin-juan-ge-zhu.top/works/book.html">-诗文总集</a> | ||
118 | <a href="https://www.qin-juan-ge-zhu.top/works/poetry.html">-诗词选</a> | 117 | <a href="https://www.qin-juan-ge-zhu.top/works/poetry.html">-诗词选</a> |
119 | <a onclick="clickit('grandson11')">-文章随笔</a> | 118 | <a onclick="clickit('grandson11')">-文章随笔</a> |
120 | <div class="grandson" id="grandson11" style="display:none"> | 119 | <div class="grandson" id="grandson11" style="display:none"> |
@@ -125,6 +124,7 @@ | |||
125 | <a href="https://www.qin-juan-ge-zhu.top/works/Napoleon.html">历史感下拿破仑</a> | 124 | <a href="https://www.qin-juan-ge-zhu.top/works/Napoleon.html">历史感下拿破仑</a> |
126 | <a href="https://www.qin-juan-ge-zhu.top/works/education.html">救救孩子!</a> | 125 | <a href="https://www.qin-juan-ge-zhu.top/works/education.html">救救孩子!</a> |
127 | </div> | 126 | </div> |
127 | <a href="https://www.qin-juan-ge-zhu.top/works/book.html">-诗文总集</a> | ||
128 | </div> | 128 | </div> |
129 | 129 | ||
130 | <div class="main-menu" id="main2" onclick="clickit('child2')">-代码世界 | 130 | <div class="main-menu" id="main2" onclick="clickit('child2')">-代码世界 |
@@ -132,28 +132,32 @@ | |||
132 | <div class="child" id="child2" style="display:none"> | 132 | <div class="child" id="child2" style="display:none"> |
133 | <a onclick="clickit('grandson21')">-美好Linux</a> | 133 | <a onclick="clickit('grandson21')">-美好Linux</a> |
134 | <div class="grandson" id="grandson21" style="display:none"> | 134 | <div class="grandson" id="grandson21" style="display:none"> |
135 | <a href="https://www.qin-juan-ge-zhu.top/code/linux/aliyun_cloud.html">-我的免费云服</a> | 135 | <a href="https://www.qin-juan-ge-zhu.top/code/linux/aliyun_cloud.html">我的免费云服</a> |
136 | <a href="https://code.qin-juan-ge-zhu.top">-VSCode Online</a> | 136 | <a href="https://www.qin-juan-ge-zhu.top/code/linux/vim.html">vim学习与配置</a> |
137 | <a href="https://www.qin-juan-ge-zhu.top/code/linux/vim.html">-vim学习与配置</a> | 137 | <a href="https://www.qin-juan-ge-zhu.top/code/linux/keymap.html">键盘映射</a> |
138 | <a href="https://www.qin-juan-ge-zhu.top/code/linux/keymap.html">-键盘映射</a> | 138 | <a href="https://www.qin-juan-ge-zhu.top/code/linux/gitserver.html">属于自己的git托管</a> |
139 | <a href="https://www.qin-juan-ge-zhu.top/code/linux/httpmyserver.html">http浏览服务器文件</a> | ||
139 | </div> | 140 | </div> |
140 | <a href="https://www.qin-juan-ge-zhu.top/code/cpppp.html">-C++ Primer | ||
141 | Plus<br> 基础知识整理</a> | ||
142 | <a href="https://www.qin-juan-ge-zhu.top/code/highlight.html">-代码块高亮与<br> 一键复制</a> | ||
143 | <a onclick="clickit('grandson22')">-一些小玩意儿</a> | 141 | <a onclick="clickit('grandson22')">-一些小玩意儿</a> |
144 | <div class="grandson" id="grandson22" style="display:none"> | 142 | <div class="grandson" id="grandson22" style="display:none"> |
145 | <a href="https://www.qin-juan-ge-zhu.top/code/index.html">-C的小零件</a> | 143 | <a href="https://www.qin-juan-ge-zhu.top/code/highlight.html">代码高亮与复制</a> |
146 | <a href="https://www.qin-juan-ge-zhu.top/code/stdc++.html">-C++九千能头文件</a> | 144 | <a href="https://www.qin-juan-ge-zhu.top/code/index.html">C的小零件</a> |
145 | <a href="https://www.qin-juan-ge-zhu.top/code/cpppp.html">C++ Primer Plus基础</a> | ||
147 | <a href="https://www.qin-juan-ge-zhu.top/code/fileop(cpp).html">C++文件操作</a> | 146 | <a href="https://www.qin-juan-ge-zhu.top/code/fileop(cpp).html">C++文件操作</a> |
148 | <a href="https://www.qin-juan-ge-zhu.top/code/experiment">大物实验MATLAB</a> | 147 | <a href="https://www.qin-juan-ge-zhu.top/code/experiment">大物实验MATLAB</a> |
149 | <a href="https://www.qin-juan-ge-zhu.top/code/zeal.html">Zeal文档下载</a> | 148 | <a href="https://www.qin-juan-ge-zhu.top/code/zeal.html">Zeal文档下载</a> |
150 | </div> | 149 | </div> |
150 | <a onclick="clickit('grandson23')">-功能网页</a> | ||
151 | <div class="grandson" id="grandson23" style="display:none"> | ||
152 | <a href="https://code.qin-juan-ge-zhu.top">VSCode Online</a> | ||
153 | <a href="https://file.qin-juan-ge-zhu.top">服务器文件浏览</a> | ||
154 | </div> | ||
151 | </div> | 155 | </div> |
152 | 156 | ||
153 | <div class="main-menu" is="main3" onclick="clickit('child3')">-读书兴味</div> | 157 | <div class="main-menu" is="main3" onclick="clickit('child3')">-读书兴味</div> |
154 | <div class="child" id="child3" style="display:none"> | 158 | <div class="child" id="child3" style="display:none"> |
155 | <a href="https://www.qin-juan-ge-zhu.top/read/on_contradictions.html">-《矛盾论》提纲</a> | 159 | <a href="https://www.qin-juan-ge-zhu.top/read/on_contradictions.html">-矛盾论提纲</a> |
156 | <a href="https://www.qin-juan-ge-zhu.top/read/nation.html">- 国家?祖国?</a> | 160 | <a href="https://www.qin-juan-ge-zhu.top/read/nation.html">-国家?祖国?</a> |
157 | </div> | 161 | </div> |
158 | <div class="main-menu" is="main4" onclick="clickit('child4')">-贪玩蓝月</div> | 162 | <div class="main-menu" is="main4" onclick="clickit('child4')">-贪玩蓝月</div> |
159 | <div class="child" id="child4" style="display:none"> | 163 | <div class="child" id="child4" style="display:none"> |
diff --git a/common/script4works.js b/common/script4works.js index 561a18c..2e47392 100644 --- a/common/script4works.js +++ b/common/script4works.js | |||
@@ -115,7 +115,6 @@ document.writeln(" </div>"); | |||
115 | document.writeln(""); | 115 | document.writeln(""); |
116 | document.writeln(" <div class=\"child\" id=\"child1\" style=\"display:none\">"); | 116 | document.writeln(" <div class=\"child\" id=\"child1\" style=\"display:none\">"); |
117 | document.writeln(" <a href=\"https://www.qin-juan-ge-zhu.top/works/index.html\">-勤倦阁记</a>"); | 117 | document.writeln(" <a href=\"https://www.qin-juan-ge-zhu.top/works/index.html\">-勤倦阁记</a>"); |
118 | document.writeln(" <a href=\"https://www.qin-juan-ge-zhu.top/works/book.html\">-诗文总集</a>"); | ||
119 | document.writeln(" <a href=\"https://www.qin-juan-ge-zhu.top/works/poetry.html\">-诗词选</a>"); | 118 | document.writeln(" <a href=\"https://www.qin-juan-ge-zhu.top/works/poetry.html\">-诗词选</a>"); |
120 | document.writeln(" <a onclick=\"clickit(\'grandson11\')\">-文章随笔</a>"); | 119 | document.writeln(" <a onclick=\"clickit(\'grandson11\')\">-文章随笔</a>"); |
121 | document.writeln(" <div class=\"grandson\" id=\"grandson11\" style=\"display:none\">"); | 120 | document.writeln(" <div class=\"grandson\" id=\"grandson11\" style=\"display:none\">"); |
@@ -126,6 +125,7 @@ document.writeln(" <a href=\"https://www.qin-juan-ge-zhu.top/work | |||
126 | document.writeln(" <a href=\"https://www.qin-juan-ge-zhu.top/works/Napoleon.html\">历史感下拿破仑</a>"); | 125 | document.writeln(" <a href=\"https://www.qin-juan-ge-zhu.top/works/Napoleon.html\">历史感下拿破仑</a>"); |
127 | document.writeln(" <a href=\"https://www.qin-juan-ge-zhu.top/works/education.html\">救救孩子!</a>"); | 126 | document.writeln(" <a href=\"https://www.qin-juan-ge-zhu.top/works/education.html\">救救孩子!</a>"); |
128 | document.writeln(" </div>"); | 127 | document.writeln(" </div>"); |
128 | document.writeln(" <a href=\"https://www.qin-juan-ge-zhu.top/works/book.html\">-诗文总集</a>"); | ||
129 | document.writeln(" </div>"); | 129 | document.writeln(" </div>"); |
130 | document.writeln(""); | 130 | document.writeln(""); |
131 | document.writeln(" <div class=\"main-menu\" id=\"main2\" onclick=\"clickit(\'child2\')\">-代码世界"); | 131 | document.writeln(" <div class=\"main-menu\" id=\"main2\" onclick=\"clickit(\'child2\')\">-代码世界"); |
@@ -133,28 +133,32 @@ document.writeln(" </div>"); | |||
133 | document.writeln(" <div class=\"child\" id=\"child2\" style=\"display:none\">"); | 133 | document.writeln(" <div class=\"child\" id=\"child2\" style=\"display:none\">"); |
134 | document.writeln(" <a onclick=\"clickit(\'grandson21\')\">-美好Linux</a>"); | 134 | document.writeln(" <a onclick=\"clickit(\'grandson21\')\">-美好Linux</a>"); |
135 | document.writeln(" <div class=\"grandson\" id=\"grandson21\" style=\"display:none\">"); | 135 | document.writeln(" <div class=\"grandson\" id=\"grandson21\" style=\"display:none\">"); |
136 | document.writeln(" <a href=\"https://www.qin-juan-ge-zhu.top/code/linux/aliyun_cloud.html\">-我的免费云服</a>"); | 136 | document.writeln(" <a href=\"https://www.qin-juan-ge-zhu.top/code/linux/aliyun_cloud.html\">我的免费云服</a>"); |
137 | document.writeln(" <a href=\"https://code.qin-juan-ge-zhu.top\">-VSCode Online</a>"); | 137 | document.writeln(" <a href=\"https://www.qin-juan-ge-zhu.top/code/linux/vim.html\">vim学习与配置</a>"); |
138 | document.writeln(" <a href=\"https://www.qin-juan-ge-zhu.top/code/linux/vim.html\">-vim学习与配置</a>"); | 138 | document.writeln(" <a href=\"https://www.qin-juan-ge-zhu.top/code/linux/keymap.html\">键盘映射</a>"); |
139 | document.writeln(" <a href=\"https://www.qin-juan-ge-zhu.top/code/linux/keymap.html\">-键盘映射</a>"); | 139 | document.writeln(" <a href=\"https://www.qin-juan-ge-zhu.top/code/linux/gitserver.html\">属于自己的git托管</a>"); |
140 | document.writeln(" <a href=\"https://www.qin-juan-ge-zhu.top/code/linux/httpmyserver.html\">http浏览服务器文件</a>"); | ||
140 | document.writeln(" </div>"); | 141 | document.writeln(" </div>"); |
141 | document.writeln(" <a href=\"https://www.qin-juan-ge-zhu.top/code/cpppp.html\">-C++ Primer"); | ||
142 | document.writeln(" Plus<br> 基础知识整理</a>"); | ||
143 | document.writeln(" <a href=\"https://www.qin-juan-ge-zhu.top/code/highlight.html\">-代码块高亮与<br> 一键复制</a>"); | ||
144 | document.writeln(" <a onclick=\"clickit(\'grandson22\')\">-一些小玩意儿</a>"); | 142 | document.writeln(" <a onclick=\"clickit(\'grandson22\')\">-一些小玩意儿</a>"); |
145 | document.writeln(" <div class=\"grandson\" id=\"grandson22\" style=\"display:none\">"); | 143 | document.writeln(" <div class=\"grandson\" id=\"grandson22\" style=\"display:none\">"); |
146 | document.writeln(" <a href=\"https://www.qin-juan-ge-zhu.top/code/index.html\">-C的小零件</a>"); | 144 | document.writeln(" <a href=\"https://www.qin-juan-ge-zhu.top/code/highlight.html\">代码高亮与复制</a>"); |
147 | document.writeln(" <a href=\"https://www.qin-juan-ge-zhu.top/code/stdc++.html\">-C++九千能头文件</a>"); | 145 | document.writeln(" <a href=\"https://www.qin-juan-ge-zhu.top/code/index.html\">C的小零件</a>"); |
146 | document.writeln(" <a href=\"https://www.qin-juan-ge-zhu.top/code/cpppp.html\">C++ Primer Plus基础</a>"); | ||
148 | document.writeln(" <a href=\"https://www.qin-juan-ge-zhu.top/code/fileop(cpp).html\">C++文件操作</a>"); | 147 | document.writeln(" <a href=\"https://www.qin-juan-ge-zhu.top/code/fileop(cpp).html\">C++文件操作</a>"); |
149 | document.writeln(" <a href=\"https://www.qin-juan-ge-zhu.top/code/experiment\">大物实验MATLAB</a>"); | 148 | document.writeln(" <a href=\"https://www.qin-juan-ge-zhu.top/code/experiment\">大物实验MATLAB</a>"); |
150 | document.writeln(" <a href=\"https://www.qin-juan-ge-zhu.top/code/zeal.html\">Zeal文档下载</a>"); | 149 | document.writeln(" <a href=\"https://www.qin-juan-ge-zhu.top/code/zeal.html\">Zeal文档下载</a>"); |
151 | document.writeln(" </div>"); | 150 | document.writeln(" </div>"); |
151 | document.writeln(" <a onclick=\"clickit(\'grandson23\')\">-功能网页</a>"); | ||
152 | document.writeln(" <div class=\"grandson\" id=\"grandson23\" style=\"display:none\">"); | ||
153 | document.writeln(" <a href=\"https://code.qin-juan-ge-zhu.top\">VSCode Online</a>"); | ||
154 | document.writeln(" <a href=\"https://file.qin-juan-ge-zhu.top\">服务器文件浏览</a>"); | ||
155 | document.writeln(" </div>"); | ||
152 | document.writeln(" </div>"); | 156 | document.writeln(" </div>"); |
153 | document.writeln(""); | 157 | document.writeln(""); |
154 | document.writeln(" <div class=\"main-menu\" is=\"main3\" onclick=\"clickit(\'child3\')\">-读书兴味</div>"); | 158 | document.writeln(" <div class=\"main-menu\" is=\"main3\" onclick=\"clickit(\'child3\')\">-读书兴味</div>"); |
155 | document.writeln(" <div class=\"child\" id=\"child3\" style=\"display:none\">"); | 159 | document.writeln(" <div class=\"child\" id=\"child3\" style=\"display:none\">"); |
156 | document.writeln(" <a href=\"https://www.qin-juan-ge-zhu.top/read/on_contradictions.html\">-《矛盾论》提纲</a>"); | 160 | document.writeln(" <a href=\"https://www.qin-juan-ge-zhu.top/read/on_contradictions.html\">-矛盾论提纲</a>"); |
157 | document.writeln(" <a href=\"https://www.qin-juan-ge-zhu.top/read/nation.html\">- 国家?祖国?</a>"); | 161 | document.writeln(" <a href=\"https://www.qin-juan-ge-zhu.top/read/nation.html\">-国家?祖国?</a>"); |
158 | document.writeln(" </div>"); | 162 | document.writeln(" </div>"); |
159 | document.writeln(" <div class=\"main-menu\" is=\"main4\" onclick=\"clickit(\'child4\')\">-贪玩蓝月</div>"); | 163 | document.writeln(" <div class=\"main-menu\" is=\"main4\" onclick=\"clickit(\'child4\')\">-贪玩蓝月</div>"); |
160 | document.writeln(" <div class=\"child\" id=\"child4\" style=\"display:none\">"); | 164 | document.writeln(" <div class=\"child\" id=\"child4\" style=\"display:none\">"); |
diff --git a/md2html.sh b/md2html.sh new file mode 100755 index 0000000..7aaee04 --- /dev/null +++ b/md2html.sh | |||
@@ -0,0 +1,28 @@ | |||
1 | #!/bin/bash | ||
2 | |||
3 | ########################################################################## | ||
4 | # File Name : md2html.sh | ||
5 | # Encoding : utf-8 | ||
6 | # Author : We-unite | ||
7 | # Email : weunite1848@gmail.com | ||
8 | # Created Time : 2023-12-15 | ||
9 | ########################################################################## | ||
10 | |||
11 | # origin是输入的第一个参数,指源文件名 | ||
12 | # src是原文件名前边加一个.,是源文件的复制 | ||
13 | # dst是原文件名的md后缀改成html | ||
14 | origin=$1 | ||
15 | src="."$origin | ||
16 | dst=${origin%.*}".html" | ||
17 | |||
18 | cp $origin $src | ||
19 | # src中所有的“```...”替换成“```”,其中...指换行前的所有内容 | ||
20 | sed -i 's/```.*$/```/g' $src | ||
21 | pandoc -s $src -o $dst | ||
22 | rm $src | ||
23 | |||
24 | sed -i '/<style/,/<\/style>/d' $dst | ||
25 | sed -i 's/<body>/<body>\n<div class="pandoc">\n<div class="main">/' $dst | ||
26 | sed -i 's/<\/body>/<script src="https:\/\/www.qin-juan-ge-zhu.top\/common\/js\/comment.js"><\/script>\n<\/div>\n<\/div>\n<\/body>/' $dst | ||
27 | sed -i 's/\t/ /g' $dst | ||
28 | sed -i 's/<\/head>/<link rel="stylesheet" href="https:\/\/www.qin-juan-ge-zhu.top\/common\/CSS\/pandoc.css">\n<script type="text\/javascript" src="https:\/\/www.qin-juan-ge-zhu.top\/common\/script4code.js"><\/script><\/head>/' $dst | ||
diff --git a/works/education.html b/works/education.html index 072e548..74b43c4 100644 --- a/works/education.html +++ b/works/education.html | |||
@@ -4,8 +4,6 @@ | |||
4 | <head> | 4 | <head> |
5 | <meta charset='UTF-8'> | 5 | <meta charset='UTF-8'> |
6 | <meta name='viewport' content='width=device-width initial-scale=1'> | 6 | <meta name='viewport' content='width=device-width initial-scale=1'> |
7 | <link href='https://fonts.loli.net/css?family=Open+Sans:400italic,700italic,700,400&subset=latin,latin-ext' | ||
8 | rel='stylesheet' type='text/css' /> | ||
9 | <script type="text/javascript" src="https://www.qin-juan-ge-zhu.top/common/script4works.js"></script> | 7 | <script type="text/javascript" src="https://www.qin-juan-ge-zhu.top/common/script4works.js"></script> |
10 | <title>救救孩子!</title> | 8 | <title>救救孩子!</title> |
11 | <!-- 评论区js --> | 9 | <!-- 评论区js --> |
diff --git a/works/poetry.html b/works/poetry.html index 848d72f..27fff26 100644 --- a/works/poetry.html +++ b/works/poetry.html | |||
@@ -6,53 +6,64 @@ | |||
6 | <meta http-equiv="X-UA-Compatible" content="IE=edge"> | 6 | <meta http-equiv="X-UA-Compatible" content="IE=edge"> |
7 | <meta name="viewport" content="width=device-width, initial-scale=1.0"> | 7 | <meta name="viewport" content="width=device-width, initial-scale=1.0"> |
8 | <script type="text/javascript" src="https://www.qin-juan-ge-zhu.top/common/script4works.js"></script> | 8 | <script type="text/javascript" src="https://www.qin-juan-ge-zhu.top/common/script4works.js"></script> |
9 | <title>词</title> | 9 | <title>桃</title> |
10 | <!-- 评论区js --> | 10 | <!-- 评论区js --> |
11 | <script src='https://www.qin-juan-ge-zhu.top/common/js/valine.min.js'></script> | 11 | <script src='https://www.qin-juan-ge-zhu.top/common/js/valine.min.js'></script> |
12 | <script src="https://cdn.jsdelivr.net/gh/flatblowfish/cave-draw/dist/cave-draw.min.js"></script> | 12 | <script src="https://cdn.jsdelivr.net/gh/flatblowfish/cave-draw/dist/cave-draw.min.js"></script> |
13 | </head> | 13 | </head> |
14 | 14 | ||
15 | <div class='typora-export os-windows'> | 15 | <body class='typora-export os-windows'> |
16 | <div class="wall"> | 16 | <div class="wall"> |
17 | <div class="background"> | 17 | <div class="background"> |
18 | <div class="box"> | 18 | <div class="box"> |
19 | <p class="title">诗词选</p> | 19 | <p class="title">诗词选</p> |
20 | <div class='md-toc' mdtype='toc'> | 20 | <div class='md-toc' mdtype='toc'> |
21 | <p class="md-toc-content" role="list"><span role="listitem" class="md-toc-item md-toc-h1" | 21 | <p class="md-toc-content" role="list"> |
22 | data-ref="n89"><a class="md-toc-inner" href="#破阵子--桃林">破阵子 桃林</a></span><br><span | 22 | <span role="listitem" class="md-toc-item md-toc-h1" data-ref="n89"><a class="md-toc-inner" |
23 | role="listitem" class="md-toc-item md-toc-h1" data-ref="n93"><a class="md-toc-inner" | 23 | href="#破阵子--桃林">破阵子 桃林</a></span><br> |
24 | href="#传韵道德">传韵道德</a></span><br><span role="listitem" class="md-toc-item md-toc-h1" | 24 | <span role="listitem" class="md-toc-item md-toc-h1" data-ref="n93"><a class="md-toc-inner" |
25 | data-ref="n96"><a class="md-toc-inner" href="#咏抗战老兵">咏抗战老兵</a></span><br><span | 25 | href="#传韵道德">传韵道德</a></span><br> |
26 | role="listitem" class="md-toc-item md-toc-h1" data-ref="n99"><a class="md-toc-inner" | 26 | <span role="listitem" class="md-toc-item md-toc-h1" data-ref="n96"><a class="md-toc-inner" |
27 | href="#凌云">凌云</a></span><br><span role="listitem" class="md-toc-item md-toc-h1" | 27 | href="#咏抗战老兵">咏抗战老兵</a></span><br> |
28 | data-ref="n102"><a class="md-toc-inner" href="#游函关故道">游函关故道</a></span><br><span | 28 | <span role="listitem" class="md-toc-item md-toc-h1" data-ref="n99"><a class="md-toc-inner" |
29 | role="listitem" class="md-toc-item md-toc-h1" data-ref="n115"><a class="md-toc-inner" | 29 | href="#凌云">凌云</a></span><br> |
30 | <span role="listitem" class="md-toc-item md-toc-h1" data-ref="n102"><a class="md-toc-inner" | ||
31 | href="#游函关故道">游函关故道</a></span><br> | ||
32 | <span role="listitem" class="md-toc-item md-toc-h1" data-ref="n115"><a class="md-toc-inner" | ||
30 | href="#捣练子-春晓踏学途乐吟">捣练子 | 33 | href="#捣练子-春晓踏学途乐吟">捣练子 |
31 | 春晓踏学途乐吟</a></span><br><span role="listitem" class="md-toc-item md-toc-h1" | 34 | 春晓踏学途乐吟</a></span><br> |
32 | data-ref="n120"><a class="md-toc-inner" href="#夜读">夜读</a></span><br><span role="listitem" | 35 | <span role="listitem" class="md-toc-item md-toc-h1" data-ref="n120"><a class="md-toc-inner" |
33 | class="md-toc-item md-toc-h1" data-ref="n123"><a class="md-toc-inner" | 36 | href="#夜读">夜读</a></span><br> |
34 | href="#游关龙逄墓">游关龙逄墓</a></span><br><span role="listitem" class="md-toc-item md-toc-h1" | 37 | <span role="listitem" class="md-toc-item md-toc-h1" data-ref="n123"><a class="md-toc-inner" |
35 | data-ref="n128"><a class="md-toc-inner" href="#南乡子">南乡子</a></span><br><span role="listitem" | 38 | href="#游关龙逄墓">游关龙逄墓</a></span><br> |
36 | class="md-toc-item md-toc-h1" data-ref="n132"><a class="md-toc-inner" | 39 | <span role="listitem" class="md-toc-item md-toc-h1" data-ref="n128"><a class="md-toc-inner" |
37 | href="#初雪">初雪</a></span><br><span role="listitem" class="md-toc-item md-toc-h1" | 40 | href="#南乡子">南乡子</a></span><br> |
38 | data-ref="n135"><a class="md-toc-inner" href="#游项城">游项城</a></span><br><span role="listitem" | 41 | <span role="listitem" class="md-toc-item md-toc-h1" data-ref="n132"><a class="md-toc-inner" |
39 | class="md-toc-item md-toc-h1" data-ref="n138"><a class="md-toc-inner" href="#鹧鸪天-夜诉衷">鹧鸪天 | 42 | href="#初雪">初雪</a></span><br> |
40 | 夜诉衷</a></span><br><span role="listitem" class="md-toc-item md-toc-h1" data-ref="n147"><a | 43 | <span role="listitem" class="md-toc-item md-toc-h1" data-ref="n135"><a class="md-toc-inner" |
41 | class="md-toc-inner" href="#早读">早读</a></span><br><span role="listitem" | 44 | href="#游项城">游项城</a></span><br> |
42 | class="md-toc-item md-toc-h1" data-ref="n153"><a class="md-toc-inner" | 45 | <span role="listitem" class="md-toc-item md-toc-h1" data-ref="n138"><a class="md-toc-inner" |
43 | href="#悼袁隆平院士">悼袁隆平院士</a></span><br><span role="listitem" class="md-toc-item md-toc-h1" | 46 | href="#鹧鸪天-夜诉衷">鹧鸪天 夜诉衷</a></span><br> |
44 | data-ref="n156"><a class="md-toc-inner" href="#赠友">赠友</a></span><br><span role="listitem" | 47 | <span role="listitem" class="md-toc-item md-toc-h1" data-ref="n147"><a class="md-toc-inner" |
45 | class="md-toc-item md-toc-h1" data-ref="n160"><a class="md-toc-inner" | 48 | href="#早读">早读</a></span><br> |
46 | href="#九九">九九</a></span><br><span role="listitem" class="md-toc-item md-toc-h1" | 49 | <span role="listitem" class="md-toc-item md-toc-h1" data-ref="n153"><a class="md-toc-inner" |
47 | data-ref="n163"><a class="md-toc-inner" href="#自贺加冠诗">自贺加冠诗</a></span><br><span | 50 | href="#悼袁隆平院士">悼袁隆平院士</a></span><br> |
48 | role="listitem" class="md-toc-item md-toc-h1" data-ref="n166"><a class="md-toc-inner" | 51 | <span role="listitem" class="md-toc-item md-toc-h1" data-ref="n156"><a class="md-toc-inner" |
49 | href="#自嘲">自嘲</a></span><br><span role="listitem" class="md-toc-item md-toc-h1" | 52 | href="#赠友">赠友</a></span><br> |
50 | data-ref="n169"><a class="md-toc-inner" href="#癸卯随笔">癸卯随笔</a></span><br><span | 53 | <span role="listitem" class="md-toc-item md-toc-h1" data-ref="n160"><a class="md-toc-inner" |
51 | role="listitem" class="md-toc-item md-toc-h1" data-ref="n172"><a class="md-toc-inner" | 54 | href="#九九">九九</a></span><br> |
55 | <span role="listitem" class="md-toc-item md-toc-h1" data-ref="n163"><a class="md-toc-inner" | ||
56 | href="#自贺加冠诗">自贺加冠诗</a></span><br> | ||
57 | <span role="listitem" class="md-toc-item md-toc-h1" data-ref="n166"><a class="md-toc-inner" | ||
58 | href="#自嘲">自嘲</a></span><br> | ||
59 | <span role="listitem" class="md-toc-item md-toc-h1" data-ref="n169"><a class="md-toc-inner" | ||
60 | href="#癸卯随笔">癸卯随笔</a></span><br> | ||
61 | <span role="listitem" class="md-toc-item md-toc-h1" data-ref="n172"><a class="md-toc-inner" | ||
52 | href="#机上作">机上作</a></span><br> | 62 | href="#机上作">机上作</a></span><br> |
63 | <span role="listitem" class="md-toc-item md-toc-h1" data-ref="n172"><a class="md-toc-inner" | ||
64 | href="#顺天三日">顺天三日</a></span><br> | ||
53 | </p> | 65 | </p> |
54 | </div> | 66 | </div> |
55 | |||
56 | 67 | ||
57 | <h1 id='破阵子--桃林'><span>破阵子 桃林</span><br></h1> | 68 | <h1 id='破阵子--桃林'><span>破阵子 桃林</span><br></h1> |
58 | <p style="text-align: center;"><span>数鸟翱翔宇内,一筝犹在云端。</span><br> | 69 | <p style="text-align: center;"><span>数鸟翱翔宇内,一筝犹在云端。</span><br> |
@@ -265,10 +276,18 @@ | |||
265 | <span>不意五国重临近,精一实愿道心存。</span><br> | 276 | <span>不意五国重临近,精一实愿道心存。</span><br> |
266 | </p> | 277 | </p> |
267 | <p class="time"><span>23.2.11</span><br></p> | 278 | <p class="time"><span>23.2.11</span><br></p> |
279 | |||
280 | <h1 id="顺天三日">顺天三日<br></h1> | ||
281 | <p style="text-align: center;"><span>夜半登车辔,晨风谒主席。</span><br> | ||
282 | <span>一哭石鼓字,三叹党人碑。</span><br> | ||
283 | <span>皇史竟谁睹,长城岂世违?</span><br> | ||
284 | <span>不闻君主业,新月笑为炊。</span><br> | ||
285 | </p> | ||
286 | <p class="time"><span>23.12.20</span><br></p> | ||
268 | <script src="https://www.qin-juan-ge-zhu.top/common/js/comment4works.js"></script> | 287 | <script src="https://www.qin-juan-ge-zhu.top/common/js/comment4works.js"></script> |
269 | </div> | 288 | </div> |
270 | </div> | 289 | </div> |
271 | </div> | 290 | </div> |
272 | </body> | 291 | </body> |
273 | 292 | ||
274 | </html> \ No newline at end of file | 293 | </html> \ No newline at end of file |