mirror of
https://github.com/LCTT/TranslateProject.git
synced 2024-12-29 21:41:00 +08:00
commit
5b2586e954
@ -0,0 +1,51 @@
|
|||||||
|
[#]: collector: (lujun9972)
|
||||||
|
[#]: translator: (qhwdw)
|
||||||
|
[#]: reviewer: (wxy)
|
||||||
|
[#]: publisher: (wxy)
|
||||||
|
[#]: url: (https://linux.cn/article-10429-1.html)
|
||||||
|
[#]: subject: (Computer Laboratory – Raspberry Pi: Lesson 0 Introduction)
|
||||||
|
[#]: via: (https://www.cl.cam.ac.uk/projects/raspberrypi/tutorials/os/introduction.html)
|
||||||
|
[#]: author: (Robert Mullins http://www.cl.cam.ac.uk/~rdm34)
|
||||||
|
|
||||||
|
计算机实验室之树莓派:课程 0 简介
|
||||||
|
======
|
||||||
|
|
||||||
|
这个课程简介不包含实践内容,但它解释了一个操作系统的基本概念、汇编代码、和其它很重要的一些基本原理。如果你想直接进入实践环节,跳过本课程并不会有什么问题。
|
||||||
|
|
||||||
|
### 1、操作系统
|
||||||
|
|
||||||
|
操作系统就是一个非常复杂的程序。它的任务就是组织安排计算机上的其它程序,包括共享计算机的时间、内存、硬件和其它资源。你可能听说过的一些比较大的桌面操作系统家族有 GNU/Linux、Mac OS X 和 Microsoft Windows。其它的设备比如电话,也需要操作系统,它可能使用的操作系统是 Android、iOS 和 [Windows Phone](https://www.cl.cam.ac.uk/projects/raspberrypi/tutorials/os/introduction.html#note1)。
|
||||||
|
|
||||||
|
由于操作系统是用来与计算机系统上的硬件进行交互的,所以它必须了解系统上硬件专有的信息。为了能让操作系统适用于各种类型的计算机,发明了 **驱动程序** 的概念。驱动程序是为了能够让操作系统与特定的硬件进行交互而添加(并可删除)到操作系统上的一小部分代码。在本课程中,我们并不涉及如何创建可删除的驱动程序,而是专注于特定的一个硬件:树莓派。
|
||||||
|
|
||||||
|
操作系统有各种不同的设计方式,在本课程中,我们只触及操作系统设计的皮毛。本课程中,我们主要专注于操作系统与各种硬件的交互部分,因为这经常是比较棘手的部分,并且也是在网络上文档和帮助最少的部分。
|
||||||
|
|
||||||
|
### 2、汇编代码
|
||||||
|
|
||||||
|
> 处理器每秒可以执行上百万的指令,但是这些指令必须要简单。
|
||||||
|
|
||||||
|
本课程几乎要完全靠汇编代码来写。汇编代码非常接近计算机的底层。计算机其实是靠一个叫处理器的设备来工作的,处理器能够执行像加法这样的简单任务,还有一组叫做 RAM 的芯片,它能够用来保存数字。当计算机通电后,处理器执行程序员给定的一系列指令,这将导致内存中的数字发生变化,以及与连接的硬件进行交互。汇编代码只是将这些机器命令转换为人类可读的文本。
|
||||||
|
|
||||||
|
常规的编程就是,程序员使用编程语言,比如 C++、Java、C#、Basic 等等来写代码,然后一个叫编译器的程序将程序员写的代码转换成汇编代码,然后进一步转换为[二进制代码](https://www.cl.cam.ac.uk/projects/raspberrypi/tutorials/os/introduction.html#note2)。二进制代码才是计算机真正能够理解的东西,但它是人类无法读取的东西。汇编代码比二进制代码好一点,至少它的命令是人类可读的,但它仍然让人很沮丧。请记住,你用汇编代码写的每个命令都是处理器可以直接认识的,因此这些命令设计的很简单,因为物理电路必须能够处理每个命令。
|
||||||
|
|
||||||
|
![Compiler process][1]
|
||||||
|
|
||||||
|
和普通编程一样,也有很多不同的汇编代码编程语言,但与普通编程不一样的是,每个汇编编程语言是面对不同的处理器的,每种处理器设计为去理解不同的语言。因此,用一个针对某种机器设计的汇编语言所写的汇编代码,是不能在其它种类的机器上运行的。很多情况下,这都是一个大灾难,因此每个程序都必须在使用它的不同种类的机器上重写一遍,但对于操作系统,这不是个问题,因为在不同的硬件上它必须得重写。尽管如此,大多数操作系统都是用 C++ 或 C 来写的,这样它们就可以很容易地在不同种类的硬件上使用,只需要重写那些必须用汇编代码来实现的部分即可。
|
||||||
|
|
||||||
|
现在,你已经准备好进入第一节课了,它是 [课程 1 OK01][2]
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: https://www.cl.cam.ac.uk/projects/raspberrypi/tutorials/os/introduction.html
|
||||||
|
|
||||||
|
作者:[Robert Mullins][a]
|
||||||
|
选题:[lujun9972][b]
|
||||||
|
译者:[qhwdw](https://github.com/qhwdw)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]: http://www.cl.cam.ac.uk/~rdm34
|
||||||
|
[b]: https://github.com/lujun9972
|
||||||
|
[1]: https://www.cl.cam.ac.uk/projects/raspberrypi/tutorials/os/images/compiling.png
|
||||||
|
[2]: https://www.cl.cam.ac.uk/projects/raspberrypi/tutorials/os/ok01.html
|
136
published/20170523 Best Websites to Download Linux Games.md
Normal file
136
published/20170523 Best Websites to Download Linux Games.md
Normal file
@ -0,0 +1,136 @@
|
|||||||
|
下载 Linux 游戏的最佳网站
|
||||||
|
======
|
||||||
|
|
||||||
|
> 新接触 Linux 游戏并想知道从哪里来 **下载 Linux 游戏**?我们列出了最好的资源,在这里你既能 **下载免费的 Linux 游戏** ,也能购买优质的 Linux 游戏。
|
||||||
|
|
||||||
|
Linux 和游戏?从前,很难想象这两者走到一起。然而随着时间流逝,很多事情都在变化。到如今,有成千上万在 Linux 上可以玩的游戏,而大公司和独立开发者们正在开发更多的游戏。
|
||||||
|
|
||||||
|
[在 Linux 上玩游戏][1] 现在已经是现实。今天我们将去看看,你在哪里可以找到 Linux 平台游戏、搜索到你喜欢的游戏。
|
||||||
|
|
||||||
|
### 在哪里来下载 Linux 游戏?
|
||||||
|
|
||||||
|
![下载 Linux 游戏的网站][2]
|
||||||
|
|
||||||
|
首先,看看你的 Linux 发行版的软件中心(如果有的话)。在这里你也能找到大量的游戏。
|
||||||
|
|
||||||
|
但是,这不意味着你应该将自己的视野局限于软件中心上。让我们来为你列出一些可以下载 Linux 游戏网站。
|
||||||
|
|
||||||
|
#### 1. Steam
|
||||||
|
|
||||||
|
如果你是老练的玩家,你应该听过 Steam。如果你还不知道的话,没错,Steam 在 Linux 上也是可用的。Steam 推荐运行在 Ubuntu 上,但是它也能运行在其它主要的发行版上。如果你真的对 Steam 很狂热,这里甚至还有一个玩 Steam 游戏的专用操作系统:[SteamOS][3]。我们在上一年 [最佳的 Linux 游戏发行版][4] 文章中提及了它。
|
||||||
|
|
||||||
|
![Steam 商店][5]
|
||||||
|
|
||||||
|
Steam 有最大的 Linux 游戏商店。当写这篇文章的时候,在 Linux 平台上,确切地说有 3487 款游戏,这真的是很多了。你可以从宽广的类型中寻找游戏。至于 [数字版权管理(DRM)][6],大多数的 Steam 游戏都有某种 DRM 。
|
||||||
|
|
||||||
|
对于使用 Steam ,要么你必须在你的 Linux 系统上安装 [Steam 客户端][7],要么使用 SteamOS。Steam 的一个优势是,在初始化安装后,对于大多数的游戏,你不需要担心依赖关系和复杂的安装过程。 Steam 客户端将为你做这些繁重的任务。
|
||||||
|
|
||||||
|
- [Steam 商店][8]
|
||||||
|
|
||||||
|
#### 2. GOG
|
||||||
|
|
||||||
|
如果你只对免 DRM 的游戏感兴趣,GOG 收集了相当多的这种游戏。此刻,GOG 在它们的库中有 1978 种免 DRM 游戏。GOG 因它大量收集了免 DRM 游戏而闻名。
|
||||||
|
|
||||||
|
![GOG 商店][9]
|
||||||
|
|
||||||
|
GOG 游戏官方支持 Ubuntu LTS 版本和 Linux Mint。所以,Ubuntu 和它的衍生版在安装它们时将没有问题。在其他发行版上安装它们时可能需要一些额外的工作,例如,你需要安装正确的依赖关系。
|
||||||
|
|
||||||
|
从 GOG 中下载游戏,你不需要额外的客户端。所有购买的游戏都可在你的账户区内找到。你可以使用你最爱的下载管理器直接下载它们。
|
||||||
|
|
||||||
|
- [GOG 商店][10]
|
||||||
|
|
||||||
|
#### 3. Humble 商店
|
||||||
|
|
||||||
|
Humble 商店是另一个你可以查找各种各样 Linux 游戏的地方。在 Humble 商店中有免 DRM 和非免 DRM 的游戏。非免 DRM 游戏通常来自 Steam。在 Humble 商店中,当前有超过 1826 款 Linux 游戏。
|
||||||
|
|
||||||
|
![Humble 商店][11]
|
||||||
|
|
||||||
|
Humble 商店因另一个原因而著名。它们有一个被称为 [Humble 独立包][12]的活动,其中打包提供了一批游戏,带有令人不可抗拒的限时优惠。关于 Humble 的另一件事是,当你购买时,你的购买金额的 10% 将捐给慈善机构。
|
||||||
|
|
||||||
|
Humble 不需要额外的客户端来下载它们的游戏。
|
||||||
|
|
||||||
|
- [Humble 商店][13]
|
||||||
|
|
||||||
|
#### 4. itch.io 商店
|
||||||
|
|
||||||
|
itch.io 是给独立数字创作者的一个开放市场,其致力于独立视频游戏。itch.io 有一些你能找到的最有趣、最独特的游戏。在 itch.io 的大多数游戏是免 DRM 的。
|
||||||
|
|
||||||
|
![itch.io 商店][14]
|
||||||
|
|
||||||
|
现今,itch.io 在它的商店中有 9514 款 Linux 平台的游戏。
|
||||||
|
|
||||||
|
itch.io 有它自己的 [客户端][15],可以轻松地下载、安装、更新和玩它们的游戏。
|
||||||
|
|
||||||
|
- [itch.io 商店][16]
|
||||||
|
|
||||||
|
#### 5. LGDB
|
||||||
|
|
||||||
|
LGDB 是 <ruby>Linux 游戏数据库<rt>Linux Game DataBase</rt></ruby>的缩写。虽然从技术上说它不是一个游戏商店,它收集有大量的 Linux 游戏,以及关于它们的各种各样的信息。每一款游戏都有你可以在哪里找到它们的链接。
|
||||||
|
|
||||||
|
![Linux 游戏数据库][17]
|
||||||
|
|
||||||
|
如今,在该数据库中有 2046 款游戏。它们也有很长的关于 [模拟器][18]、[工具][19] 和 [游戏引擎][20] 的列表。
|
||||||
|
|
||||||
|
- [LGDB][21]
|
||||||
|
|
||||||
|
#### 6. Game Jolt 商店
|
||||||
|
|
||||||
|
Game Jolt 有一个非常可观的集合,在它的库藏中大约有 5000 个 Linux 独立游戏。
|
||||||
|
|
||||||
|
![Game Jolt 商店][22]
|
||||||
|
|
||||||
|
Game Jolt 有一个(预览版)[客户端][23],可用于轻松地下载、安装、更新和玩游戏。
|
||||||
|
|
||||||
|
- [Game Jolt 商店][24]
|
||||||
|
|
||||||
|
### 其他
|
||||||
|
|
||||||
|
有很多其他的销售 Linux 游戏的商店。也有很多你能找到免费游戏的地方。这是它们中的两个:
|
||||||
|
|
||||||
|
* [Bundle Stars][25]:当前有 814 个 Linux 游戏和 31 个游戏包。
|
||||||
|
* [GamersGate][26]:现在有 595 个 Linux 游戏。既有免 DRM 的,也有非免 DRM 的。
|
||||||
|
|
||||||
|
#### 应用商店、软件中心 & 软件库
|
||||||
|
|
||||||
|
Linux 发行版有它们自己的应用商店或软件库。尽管不是很多,但是在这里你也能找到各种各样的游戏。
|
||||||
|
|
||||||
|
今天到此为止。你知道这里有这么多 Linux 上可玩的游戏吗?你使用一些其他的网站来下载 Linux 游戏吗?与我们分享你的收藏。
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: https://itsfoss.com/download-linux-games/
|
||||||
|
|
||||||
|
作者:[Munif Tanjim][a]
|
||||||
|
译者:[robsean](https://github.com/robsean)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]: https://itsfoss.com/author/munif/
|
||||||
|
[1]:https://linux.cn/article-7316-1.html
|
||||||
|
[2]:https://itsfoss.com/wp-content/uploads/2017/05/download-linux-games-800x450.jpg
|
||||||
|
[3]:http://store.steampowered.com/steamos/
|
||||||
|
[4]:https://itsfoss.com/linux-gaming-distributions/
|
||||||
|
[5]:https://itsfoss.com/wp-content/uploads/2017/05/Steam-Store-800x382.jpg
|
||||||
|
[6]:https://www.wikiwand.com/en/Digital_rights_management
|
||||||
|
[7]:http://store.steampowered.com/about/
|
||||||
|
[8]:http://store.steampowered.com/linux
|
||||||
|
[9]:https://itsfoss.com/wp-content/uploads/2017/05/GOG-Store-800x366.jpg
|
||||||
|
[10]:https://www.gog.com/games?system=lin_mint,lin_ubuntu
|
||||||
|
[11]:https://itsfoss.com/wp-content/uploads/2017/05/The-Humble-Store-800x393.jpg
|
||||||
|
[12]:https://www.humblebundle.com/?partner=itsfoss
|
||||||
|
[13]:https://www.humblebundle.com/store?partner=itsfoss
|
||||||
|
[14]:https://itsfoss.com/wp-content/uploads/2017/05/itch.io-Store-800x485.jpg
|
||||||
|
[15]:https://itch.io/app
|
||||||
|
[16]:https://itch.io/games/platform-linux
|
||||||
|
[17]:https://itsfoss.com/wp-content/uploads/2017/05/LGDB-800x304.jpg
|
||||||
|
[18]:https://lgdb.org/emulators
|
||||||
|
[19]:https://lgdb.org/tools
|
||||||
|
[20]:https://lgdb.org/engines
|
||||||
|
[21]:https://lgdb.org/games
|
||||||
|
[22]:https://itsfoss.com/wp-content/uploads/2017/05/GameJolt-Store-800x357.jpg
|
||||||
|
[23]:http://gamejolt.com/client
|
||||||
|
[24]:http://gamejolt.com/games/best?os=linux
|
||||||
|
[25]:https://www.bundlestars.com/en/games?page=1&platforms=Linux
|
||||||
|
[26]:https://www.gamersgate.com/games?state=available
|
||||||
|
[27]:https://itsfoss.com/linux-gaming-problems/
|
@ -0,0 +1,112 @@
|
|||||||
|
如何用 Gonimo 创建一个免费的婴儿监视系统
|
||||||
|
======
|
||||||
|
> 当你可以用两个设备、浏览器和网络连接就能免费搭建一个婴儿监视器时,谁还会花钱去买呢?
|
||||||
|
|
||||||
|
![](https://opensource.com/sites/default/files/styles/image-full-size/public/lead-images/baby.png?itok=7jyDs9vE)
|
||||||
|
|
||||||
|
新父母和准父母很快就会知道将会有一个既长且昂贵的新生儿所需设备的清单,清单中的首位是一个婴儿监视器,借此他们可以在做其他事情时照看自己的婴儿,但这儿有一件不必消耗你的婴儿经费的设备,Gonimo 是一个可以将现有的设备转换成婴儿监控系统的自由开源解决方案,附近大型婴儿用品商店的过道中有数以千计的其他必备或时尚物品,就可以为其中某一个腾出一些婴儿的预算。
|
||||||
|
|
||||||
|
Gonimo 诞生时,它的开发者,一个有双胞胎的开源粉丝,发现现有选择存在问题:
|
||||||
|
|
||||||
|
* 现有硬件婴儿监视器价格昂贵,使用范围有限,需要您带着额外的设备。
|
||||||
|
* 虽然有移动监控应用程序,但大多数现有的 iOS / Android 婴儿监控应用程序都不可靠且不安全,不太容易找到开源产品。
|
||||||
|
* 如果您有两个小孩(例如双胞胎),您将需要两台监视器,使您的成本翻倍。
|
||||||
|
|
||||||
|
Gonimo 是作为一个解决典型的监视器的缺点的开源解决方案而创建的:
|
||||||
|
|
||||||
|
* 昂贵?不,它是免费的!
|
||||||
|
* 使用范围有限?不,它适用于互联网 / WiFi,无论您身在何处。
|
||||||
|
* 下载并安装应用程序?噢不,它适用于您现有的网络浏览器。
|
||||||
|
* 购买新设备?不用,你可以使用任何笔记本电脑、手机或平板电脑与网络浏览器和麦克风和/或相机。
|
||||||
|
|
||||||
|
(注意:遗憾的是,Apple iOS 设备尚不支持,但预计很快就会发生变化 —— 请继续阅读,了解如何帮实现这一目标。)
|
||||||
|
|
||||||
|
### 开始
|
||||||
|
|
||||||
|
将您的设备转换为婴儿监视器很容易。从您设备的浏览器(理想情况下为 Chrome),访问 [gonimo.com][1] 并单击 “Start baby monitor” 以访问该 Web 应用程序。
|
||||||
|
|
||||||
|
1、创建家庭:在首次启动的屏幕上,你会看到一只可爱的兔子在地球上奔跑。这是您创建新家庭的地方。单击 “+” 按钮并接受随机生成的姓氏或键入您自己的选择。
|
||||||
|
|
||||||
|
![Start screen][3]
|
||||||
|
|
||||||
|
*从开始屏幕创建一个新家庭*
|
||||||
|
|
||||||
|
2、邀请设备:建立完家庭以后,下个屏幕将指示您邀请其他设备加入你的 Gonimo 家庭。您可以通过电子邮件直接发送一次性邀请链接,也可以将其复制并粘贴到消息中。对其他设备,只需打开该链接并接受邀请。对您要邀请的任何其他设备重复此过程。您的设备现在属于同一家庭,可以作为一个完全正常工作的婴儿监视器系统进行配合。
|
||||||
|
|
||||||
|
![Invite screen][5]
|
||||||
|
|
||||||
|
*邀请家庭成员*
|
||||||
|
|
||||||
|
3、启动婴儿站流:接下来,通过转到 [Gonimo 主屏][6],点击带有奶嘴的按钮,并允许网络浏览器访问设备的麦克风和摄像头,选择将婴儿的音频和视频流式传输到父母的设备。调整相机以指向宝宝的床,或关闭它以节省设备电池(音频仍将流式传输)。点击“Start”。该流现在处于活动状态。
|
||||||
|
|
||||||
|
![Select baby station][8]
|
||||||
|
|
||||||
|
*选择婴儿站*
|
||||||
|
|
||||||
|
![Press Start][10]
|
||||||
|
|
||||||
|
*按“Start”开始以流式传输视频*
|
||||||
|
|
||||||
|
4、连接到父母站流:要查看婴儿站流,请转到 Gonimo 家族中的另外的设备 —— 即父母站。点击 Gonimo 主屏幕上的 “Parent” 按钮。您将看到该系列中所有设备的列表;旁边有一个闪动的“Connect”按钮的是活跃的婴儿站。选择“Connect”,您可以通过点对点音频/视频流看到和听到您的宝宝。音量条为传输的音频流提供可视化。
|
||||||
|
|
||||||
|
![Select parent station][12]
|
||||||
|
|
||||||
|
*选择父母站*
|
||||||
|
|
||||||
|
![Press Connect][14]
|
||||||
|
|
||||||
|
*按下“Connect”开始观看婴儿流。*
|
||||||
|
|
||||||
|
5、恭喜!您已成功将设备直接通过网络浏览器转换为婴儿监视器,无需下载或安装任何应用程序!
|
||||||
|
|
||||||
|
有关重命名设备,从系列中删除设备或删除系列的详细信息和详细说明,请查看 gonimo.com 的[视频教程][15]。
|
||||||
|
|
||||||
|
### 家庭系统的灵活性
|
||||||
|
|
||||||
|
Gonimo 的优势之一是其基于家庭的系统,它为即使在商业 Android 或 iOS 应用中也无法提供的各种情况提供了极大的灵活性。要深入了解这些功能,我们假设您创建了一个由三个设备组成的家庭系统。
|
||||||
|
|
||||||
|
* 多婴儿:如果你想留意你在两个不同房间睡觉的两个小孩怎么办?在每个孩子的房间放一个设备,并将其设置为婴儿站。第三个设备将充当父母站,您可以在其上连接到两个流并通过分屏查看您的幼儿。您甚至可以通过向该家庭系统提供更多设备,并将其设置为婴儿站来将此用例扩展到两个以上的婴儿站。只要您的父母站连接到第一个婴儿站,请单击左上角的后退箭头返回“设备概述”屏幕。现在您可以连接到第二个(以及依次为第三个、第四个、第五个和第五个等)设备,并自动建立分屏。酷!
|
||||||
|
* 多父母:如果爸爸想在他上班的时候看孩子怎么办?只需邀请第四个设备(例如,他的办公室 PC )到家庭并将其设置为父母站。父母双方都可以通过他们自己的设备同时检查他们的孩子,甚至可以独立地选择他们希望连接的孩子。
|
||||||
|
* 多家庭:单个设备也可以是几个家庭系统的一部分。当您的婴儿站与您一起时,如平板电脑,您经常访问亲戚或朋友时,这非常有用。为“奶奶的房子”或“约翰叔叔的房子”创建另一个家庭,其中包括您的婴儿站设备与奶奶或约翰叔叔的设备配对。您可以随时通过婴儿站设备的 Gonimo 主屏幕在这些家庭中切换婴儿站设备。
|
||||||
|
|
||||||
|
### 想要参加吗?
|
||||||
|
|
||||||
|
Gonimo 团队喜欢开源。代码来自社区,代码用于社区。Gonimo 的用户对我们非常重要,但它们只是 Gonimo 故事的一部分。幕后有创意的大脑是创造出色婴儿监视器体验的关键。
|
||||||
|
|
||||||
|
目前我们特别需要那些愿意成为 iOS 11 测试者的人的帮助,因为 Apple 在 iOS 11 中对 WebRTC 的支持意味着我们最终将能够支持 iOS 设备。如果可以,请帮助我们实现这个令人赞叹的里程碑。
|
||||||
|
|
||||||
|
如果您了解 Haskell 或想要了解它,您可以查看 [GitHub 上我们的代码][16]。欢迎发起拉取请求、审查代码以及提出问题。
|
||||||
|
|
||||||
|
最后,请通过向新父母和开源世界宣传 Gonimo 婴儿监视器是易于使用并且便携的。
|
||||||
|
|
||||||
|
### 关于作者
|
||||||
|
|
||||||
|
Robert Klotzner:我是双胞胎的父亲,一个程序员。当我听到普通人可以给计算机编程时,我买了一本 1200 页的关于 C++ 的书开始学习,我当时才十五岁。我坚持用 C++ 相当长的一段时间,学习了 Java 又回归到 C++,学习了 D、Python 等等,最终学习了 Haskell。我喜欢 Haskell 是因为它丰富的类型系统,这几乎可以避免我书写错误的代码。强壮的静态类型和性能让我爱上了 C++……
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: https://opensource.com/article/17/9/gonimo
|
||||||
|
|
||||||
|
作者:[Robert Klotzner][a]
|
||||||
|
译者:[lintaov587](https://github.com/lintaov587)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]:https://opensource.com/users/robert-klotzner
|
||||||
|
[1]:https://gonimo.com/
|
||||||
|
[2]:/file/371256
|
||||||
|
[3]:https://opensource.com/sites/default/files/u128651/start-screen.png (Start screen)
|
||||||
|
[4]:/file/371236
|
||||||
|
[5]:https://opensource.com/sites/default/files/u128651/invite-screen.png (Invite screen)
|
||||||
|
[6]:https://app.gonimo.com/
|
||||||
|
[7]:/file/371231
|
||||||
|
[8]:https://opensource.com/sites/default/files/u128651/baby-select.png (Select baby station)
|
||||||
|
[9]:/file/371226
|
||||||
|
[10]:https://opensource.com/sites/default/files/u128651/baby-screen.png (Press Start)
|
||||||
|
[11]:/file/371251
|
||||||
|
[12]:https://opensource.com/sites/default/files/u128651/parent-select.png (Select parent station)
|
||||||
|
[13]:/file/371241
|
||||||
|
[14]:https://opensource.com/sites/default/files/u128651/parent-screen.png (Press Connect)
|
||||||
|
[15]:https://gonimo.com/index.php#intro
|
||||||
|
[16]:https://github.com/gonimo/gonimo
|
316
published/20171227 YAML- probably not so great after all.md
Normal file
316
published/20171227 YAML- probably not so great after all.md
Normal file
@ -0,0 +1,316 @@
|
|||||||
|
[#]: collector: (lujun9972)
|
||||||
|
[#]: translator: (MjSeven)
|
||||||
|
[#]: reviewer: (wxy)
|
||||||
|
[#]: publisher: (wxy)
|
||||||
|
[#]: url: (https://linux.cn/article-10423-1.html)
|
||||||
|
[#]: subject: (YAML: probably not so great after all)
|
||||||
|
[#]: via: (https://arp242.net/weblog/yaml_probably_not_so_great_after_all.html)
|
||||||
|
[#]: author: (Martin Tournoij https://arp242.net/)
|
||||||
|
|
||||||
|
YAML:可能并不是那么完美
|
||||||
|
======
|
||||||
|
|
||||||
|
我之前写过[为什么将 JSON 用于人类可编辑的配置文件是一个坏主意][1],今天我们将讨论 YAML 格式的一些常见问题。
|
||||||
|
|
||||||
|
### 默认情况下不安全
|
||||||
|
|
||||||
|
YAML 默认是不安全的。加载用户提供的(不可信的)YAML 字符串需要仔细考虑。
|
||||||
|
|
||||||
|
```
|
||||||
|
!!python/object/apply:os.system
|
||||||
|
args: ['ls /']
|
||||||
|
```
|
||||||
|
|
||||||
|
用 `print(yaml.load(open('a.yaml')))` 运行它,应该给你这样的东西:
|
||||||
|
|
||||||
|
```
|
||||||
|
bin etc lib lost+found opt root sbin tmp var sys
|
||||||
|
boot dev efi home lib64 mnt proc run srv usr
|
||||||
|
0
|
||||||
|
```
|
||||||
|
|
||||||
|
许多其他语言(包括 Ruby 和 PHP [^1] )默认情况下也不安全(LCTT 译注:这里应该说的是解析 yaml)。[在 GitHub 上搜索 yaml.load][2] 会得到惊人的 280 万个结果,而 [yaml.safe_load][3] 只能得到 26000 个结果。
|
||||||
|
|
||||||
|
提个醒,很多这样的 `yaml.load()` 都工作的很好,在配置文件中加载 `yaml.load()` 通常没问题,因为它通常(虽然并不总是!)来自“可靠源”,而且很多都来自静态的 YAML 测试文件。但是,人们还是不禁怀疑在这 280 万个结果中隐藏了多少漏洞。
|
||||||
|
|
||||||
|
这不是一个理论问题。在 2013 年,[正是由于这个问题,所有的 Ruby on Rails 应用程序都被发现易受][4]远程代码执行攻击。
|
||||||
|
|
||||||
|
有人可能会反驳说这不是 YAML 格式的错误,而是那些库实现错误的的问题,但似乎大多数库默认不是安全的(特别是动态语言),所以事实上这是 YAML 的一个问题。
|
||||||
|
|
||||||
|
有些人可能会反驳认为修复它就像用 `safe_load()` 替换 `load()` 一样容易,但是很多人都没有意识到这个问题,即使你知道它,它也是很容易忘记的事情之一。这是非常糟糕的 API 设计。
|
||||||
|
|
||||||
|
### 可能很难编辑,特别是对于大文件
|
||||||
|
|
||||||
|
YAML 文件可能很难编辑,随着文件变大,这个难度会快速增大。
|
||||||
|
|
||||||
|
一个很好的例子是 Ruby on Rails 的本地化翻译文件。例如:
|
||||||
|
|
||||||
|
```
|
||||||
|
en:
|
||||||
|
formtastic:
|
||||||
|
labels:
|
||||||
|
title: "Title" # Default global value
|
||||||
|
article:
|
||||||
|
body: "Article content"
|
||||||
|
post:
|
||||||
|
new:
|
||||||
|
title: "Choose a title..."
|
||||||
|
body: "Write something..."
|
||||||
|
edit:
|
||||||
|
title: "Edit title"
|
||||||
|
body: "Edit body"
|
||||||
|
```
|
||||||
|
|
||||||
|
看起来不错,对吧?但是如果这个文件有 100 行怎么办?或者 1,000 行?在文件中很难看到 “where”,因为它可能在屏幕外。你需要向上滚动,但是你需要跟踪缩进,即使遵循缩进指南也很难,特别是因为 2 个空格缩进是常态而且 [制表符缩进被禁止][5] [^2] 。
|
||||||
|
|
||||||
|
不小心缩进出错通常不算错误,它通常只是反序列化为你不想要的东西。这样只能祝你调试快乐!
|
||||||
|
|
||||||
|
我已经愉快地编写 Python 长达十多年,所以我已经习惯了显眼的空白,但有时候我仍在和 YAML 抗争。在 Python 中,虽然没有那种长达几页的函数,但数据或配置文件的长度没有这种自然限制,这就带来了缺点和损失了清晰度。
|
||||||
|
|
||||||
|
对于小文件,这不是问题,但它确实无法很好地扩展到较大的文件,特别是如果你以后想编辑它们的话。
|
||||||
|
|
||||||
|
### 这非常复杂
|
||||||
|
|
||||||
|
在浏览一个基本的例子时,YAML 看似“简单”和“显而易见”,但事实证明并非如此。[YAML 规范][6]有 23449 个单词,为了比较,[TOML][7] 有 3339 个单词,[Json][8] 有 1969 个单词,[XML][9] 有 20603 个单词。
|
||||||
|
|
||||||
|
我们中有谁读过全部规范吗?有谁读过并理解了全部?谁阅读过,理解进而**记住**所有这些?
|
||||||
|
|
||||||
|
例如,你知道[在 YAML 中编写多行字符串有 9 种方法][10]吗?并且它们具有细微的不同行为。
|
||||||
|
|
||||||
|
是的 :-/
|
||||||
|
|
||||||
|
如果你看一下[那篇文章的修订历史][11],它就会变得更加有趣,因为文章的作者发现了越来越多的方法可以实现这一点,以及更多的细微之处。
|
||||||
|
|
||||||
|
它从预览开始告诉我们 YAML 规范,它表明(强调我的):
|
||||||
|
|
||||||
|
> 本节简要介绍了 YAML 的表达能力。**预计初次阅读的人不可能理解所有的例子**。相反,这些选择用作该规范其余部分的动机。
|
||||||
|
|
||||||
|
#### 令人惊讶的行为
|
||||||
|
|
||||||
|
以下会解析成什么([Colm O’Connor][12] 提供的例子):
|
||||||
|
|
||||||
|
```
|
||||||
|
- Don Corleone: Do you have faith in my judgment?
|
||||||
|
- Clemenza: Yes
|
||||||
|
- Don Corleone: Do I have your loyalty?
|
||||||
|
```
|
||||||
|
|
||||||
|
结果为:
|
||||||
|
|
||||||
|
```
|
||||||
|
[
|
||||||
|
{'Don Corleone': 'Do you have faith in my judgment?'},
|
||||||
|
{'Clemenza': True},
|
||||||
|
{'Don Corleone': 'Do I have your loyalty?'}
|
||||||
|
]
|
||||||
|
```
|
||||||
|
|
||||||
|
那么这个呢:
|
||||||
|
|
||||||
|
```
|
||||||
|
python: 3.5.3
|
||||||
|
postgres: 9.3
|
||||||
|
```
|
||||||
|
|
||||||
|
`3.5.3` 被识别为字符串,但 `9.3` 被识别为数字而不是字符串:
|
||||||
|
|
||||||
|
```
|
||||||
|
{'python': '3.5.3', 'postgres': 9.3}
|
||||||
|
```
|
||||||
|
|
||||||
|
这个呢:
|
||||||
|
|
||||||
|
```
|
||||||
|
Effenaar: Eindhoven
|
||||||
|
013: Tilburg
|
||||||
|
```
|
||||||
|
|
||||||
|
`013` 是<ruby>蒂尔堡<rt>Tilburg<rt></ruby>的一个流行音乐场地,但 YAML 会告诉你错误答案,因为它被解析为八进制数字:
|
||||||
|
|
||||||
|
```
|
||||||
|
{11: 'Tilburg', 'Effenaar': 'Eindhoven'}
|
||||||
|
```
|
||||||
|
|
||||||
|
所有这一切,以及更多,就是为什么许多经验丰富的 YAMLer 经常会将所有字符串用引号引起来的原因,即使它不是严格要求。许多人不使用引号,而且很容易忘记,特别是如果文件的其余部分(可能由其他人编写)不使用引号。
|
||||||
|
|
||||||
|
### 它不方便
|
||||||
|
|
||||||
|
因为它太复杂了,它所声称的可移植性被夸大了。例如,考虑以下这个从 YAML 规范中获取的示例:
|
||||||
|
|
||||||
|
```
|
||||||
|
? - Detroit Tigers
|
||||||
|
- Chicago cubs
|
||||||
|
:
|
||||||
|
- 2001-07-23
|
||||||
|
|
||||||
|
? [ New York Yankees,
|
||||||
|
Atlanta Braves ]
|
||||||
|
: [ 2001-07-02, 2001-08-12,
|
||||||
|
2001-08-14 ]
|
||||||
|
```
|
||||||
|
|
||||||
|
抛开大多数读者可能甚至不知道这是在做什么之外,请尝试使用 PyYAML 在 Python 中解析它:
|
||||||
|
|
||||||
|
```
|
||||||
|
yaml.constructor.ConstructorError: while constructing a mapping
|
||||||
|
in "a.yaml", line 1, column 1
|
||||||
|
found unhashable key
|
||||||
|
in "a.yaml", line 1, column 3
|
||||||
|
```
|
||||||
|
|
||||||
|
在 Ruby 中,它可以工作:
|
||||||
|
|
||||||
|
```
|
||||||
|
{
|
||||||
|
["Detroit Tigers", "Chicago cubs"] => [
|
||||||
|
#<Date: 2001-07-23 ((2452114j,0s,0n),+0s,2299161j)>
|
||||||
|
],
|
||||||
|
["New York Yankees", "Atlanta Braves"] => [
|
||||||
|
#<Date: 2001-07-02 ((2452093j,0s,0n),+0s,2299161j)>,
|
||||||
|
#<Date: 2001-08-12 ((2452134j,0s,0n),+0s,2299161j)>,
|
||||||
|
#<Date: 2001-08-14 ((2452136j,0s,0n),+0s,2299161j)>
|
||||||
|
]
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
这个原因是你不能在 Python 中使用列表作为一个字典的键:
|
||||||
|
|
||||||
|
```
|
||||||
|
>>> {['a']: 'zxc'}
|
||||||
|
Traceback (most recent call last):
|
||||||
|
File "<stdin>", line 1, in <module>
|
||||||
|
TypeError: unhashable type: 'list'
|
||||||
|
```
|
||||||
|
|
||||||
|
而这种限制并不是 Python 特有的,PHP、JavaScript 和 Go 等常用语言都有此限制。
|
||||||
|
|
||||||
|
因此,在 YAML 文件中使用这种语法,你将无法在大多数语言中解析它。
|
||||||
|
|
||||||
|
这是另一个从 YAML 规范的示例部分中获取的:
|
||||||
|
|
||||||
|
```
|
||||||
|
# Ranking of 1998 home runs
|
||||||
|
---
|
||||||
|
- Mark McGwire
|
||||||
|
- Sammy Sosa
|
||||||
|
- Ken Griffey
|
||||||
|
|
||||||
|
# Team ranking
|
||||||
|
---
|
||||||
|
- Chicago Cubs
|
||||||
|
- St Louis Cardinals
|
||||||
|
```
|
||||||
|
|
||||||
|
Python 会输出:
|
||||||
|
|
||||||
|
```
|
||||||
|
yaml.composer.ComposerError: expected a single document in the stream
|
||||||
|
in "a.yaml", line 3, column 1
|
||||||
|
but found another document
|
||||||
|
in "a.yaml", line 8, column 1
|
||||||
|
```
|
||||||
|
|
||||||
|
然而 Ruby 输出:
|
||||||
|
|
||||||
|
```
|
||||||
|
["Mark McGwire", "Sammy Sosa", "Ken Griffey"]
|
||||||
|
```
|
||||||
|
|
||||||
|
原因是单个文件中有多个 YAML 文档(`---` 意味开始一个新文档)。在 Python 中,有一个 `load_all` 函数来解析所有文档,而 Ruby 的 `load()` 只是加载第一个文档,据我所知,它没有办法加载多个文档。
|
||||||
|
|
||||||
|
[在实现之间存在很多不兼容][13]。
|
||||||
|
|
||||||
|
### 目标实现了吗?
|
||||||
|
|
||||||
|
规范说明:
|
||||||
|
|
||||||
|
> YAML 的设计目标是降低优先级:
|
||||||
|
>
|
||||||
|
> 1. YAML 很容易被人类阅读。
|
||||||
|
> 2. YAML 数据在编程语言之间是可移植的。
|
||||||
|
> 3. YAML 匹配敏捷语言的原生数据结构。
|
||||||
|
> 4. YAML 有一个一致的模型来支持通用工具。
|
||||||
|
> 5. YAML 支持一次性处理。
|
||||||
|
> 6. YAML 具有表现力和可扩展性。
|
||||||
|
> 7. YAML 易于实现和使用。
|
||||||
|
|
||||||
|
那么它做的如何呢?
|
||||||
|
|
||||||
|
> YAML 很容易被人类阅读。
|
||||||
|
|
||||||
|
只有坚持一小部分子集时才有效。完整的规则集很复杂 —— 远远超过 XML 或 JSON。
|
||||||
|
|
||||||
|
> YAML 数据在编程语言之间是可移植的。
|
||||||
|
|
||||||
|
事实并非如此,因为创建常见语言不支持的结构太容易了。
|
||||||
|
|
||||||
|
> YAML 匹配敏捷语言的原生数据结构。
|
||||||
|
|
||||||
|
参见上面。另外,为什么只支持敏捷(或动态)语言?其他语言呢?
|
||||||
|
|
||||||
|
> YAML 有一个一致的模型来支持通用工具。
|
||||||
|
|
||||||
|
我甚至不确定这意味着什么,我找不到任何详细说明。
|
||||||
|
|
||||||
|
> YAML 支持一次性处理。
|
||||||
|
|
||||||
|
这点我接受。
|
||||||
|
|
||||||
|
> YAML 具有表现力和可扩展性。
|
||||||
|
|
||||||
|
嗯,是的,但它太富有表现力(例如太复杂)。
|
||||||
|
|
||||||
|
> YAML 易于实现和使用。
|
||||||
|
|
||||||
|
```
|
||||||
|
$ cat `ls -1 ~/gocode/src/github.com/go-yaml/yaml/*.go | grep -v _test` | wc -l
|
||||||
|
9247
|
||||||
|
|
||||||
|
$ cat /usr/lib/python3.5/site-packages/yaml/*.py | wc -l
|
||||||
|
5713
|
||||||
|
```
|
||||||
|
|
||||||
|
### 结论
|
||||||
|
|
||||||
|
不要误解我的意思,并不是说 YAML 很糟糕 —— 它肯定不像[使用 JSON 那么多的问题][1] —— 但它也不是非常好。有一些一开始并不明显的缺点和惊喜,还有许多更好的替代品,如 [TOML][7] 和其他更专业的格式。
|
||||||
|
|
||||||
|
就个人而言,当我有选择时,我不太可能再次使用它。
|
||||||
|
|
||||||
|
如果你必须使用 YAML,那么我建议你使用 [StrictYAML][14],它会删除一些(虽然不是全部)比较麻烦的部分。
|
||||||
|
|
||||||
|
### 反馈
|
||||||
|
|
||||||
|
你可以发送电子邮件至 [martin@arp242.net][15] 或[创建 GitHub issue][16] 以获取反馈、问题等。
|
||||||
|
|
||||||
|
[^1]: 在 PHP 中你需要修改一个 INI 设置来获得安全的行为,不能只是调用像 `yaml_safe()` 这样的东西。PHP 想尽办法让愚蠢的东西越发愚蠢。干得漂亮!
|
||||||
|
[^2]: 不要在这里做空格与制表符之争,如果这里可以用制表符的话,我可以(临时)增加制表符宽度来使它更易读——这是制表符的一种用途。
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: https://arp242.net/weblog/yaml_probably_not_so_great_after_all.html
|
||||||
|
|
||||||
|
作者:[Martin Tournoij][a]
|
||||||
|
选题:[lujun9972][b]
|
||||||
|
译者:[MjSeven](https://github.com/MjSeven)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]: https://arp242.net/
|
||||||
|
[b]: https://github.com/lujun9972
|
||||||
|
[1]: http://arp242.net/weblog/JSON_as_configuration_files-_please_dont.html
|
||||||
|
[2]: https://github.com/search?q=yaml.load&type=Code&utf8=%E2%9C%93
|
||||||
|
[3]: https://github.com/search?q=yaml.safe_load&type=Code&utf8=%E2%9C%93
|
||||||
|
[4]: https://www.sitepoint.com/anatomy-of-an-exploit-an-in-depth-look-at-the-rails-yaml-vulnerability/
|
||||||
|
[5]: http://www.yaml.org/faq.html
|
||||||
|
[6]: http://yaml.org/spec/1.2/spec.pdf
|
||||||
|
[7]: https://github.com/toml-lang/toml
|
||||||
|
[8]: http://www.ecma-international.org/publications/files/ECMA-ST/ECMA-404.pdf
|
||||||
|
[9]: https://www.w3.org/TR/REC-xml/
|
||||||
|
[10]: http://stackoverflow.com/a/21699210/660921
|
||||||
|
[11]: http://stackoverflow.com/posts/21699210/revisions
|
||||||
|
[12]: https://github.com/crdoconnor/strictyaml/blob/master/FAQ.rst#what-is-wrong-with-implicit-typing
|
||||||
|
[13]: https://github.com/cblp/yaml-sucks
|
||||||
|
[14]: https://github.com/crdoconnor/strictyaml
|
||||||
|
[15]: mailto:martin@arp242.net
|
||||||
|
[16]: https://github.com/Carpetsmoker/arp242.net/issues/new
|
@ -0,0 +1,90 @@
|
|||||||
|
5 款 Linux 街机游戏
|
||||||
|
================
|
||||||
|
|
||||||
|
![](https://opensource.com/sites/default/files/styles/image-full-size/public/lead-images/arcade_game_gaming.jpg?itok=84Rjk_32)
|
||||||
|
|
||||||
|
长久以来,游戏都是 Linux 的软肋。近些年,Steam、GOG 等游戏发布平台上不少商业游戏都开始支持 Linux,这对于 Linux 的游戏生态来说是件好事,但是我们能在这些平台上玩到的游戏通常是不开源的商业作品。当然,这些游戏在一个开源的操作系统上运行,但对于一个开源提倡者来说这似乎还不够纯粹。
|
||||||
|
|
||||||
|
那么,我们能找到既自由开源又能给玩家带来完整游戏体验的优质游戏吗?当然!虽然绝大多数的开源游戏很难和 3A 商业游戏大作竞争,但仍然有不少各种类型的开源游戏,不仅内容有趣而且直接可以通过几大 Linux 发行版本库中直接安装。
|
||||||
|
|
||||||
|
本文首先介绍 Linux 开源游戏中的街机类型游戏,在之后的文章中,我将介绍桌面和卡牌游戏,解谜游戏,竞速游戏,以及策略模拟游戏。
|
||||||
|
|
||||||
|
### <ruby>太空危机<rt>AstroMenace</rt></ruby>
|
||||||
|
|
||||||
|
![](https://opensource.com/sites/default/files/u128651/astromenace.png)
|
||||||
|
|
||||||
|
[太空危机][3] 是一个近现代太空背景下的滚动页面射击游戏。开发初期它是一个闭源游戏,但它的代码和素材而后以开源许可证发布了。游戏玩法和大多数此类游戏大同小异,但它有质量极高的 3D 画面。飞船和武器升级可以通过击杀敌人所获得的点数购买。游戏的难度可以选择,因此适合新手以及想要追求挑战的硬核玩家。
|
||||||
|
|
||||||
|
安装太空危机,你只需要在终端下运行以下指令:
|
||||||
|
|
||||||
|
- Fedora 用户: `dnf install astromenace`
|
||||||
|
- Debian/Ubuntu 用户: `apt install astromenace`
|
||||||
|
|
||||||
|
### <ruby>坦克战役<rt>Battle Tanks</rt></ruby>
|
||||||
|
|
||||||
|
![](https://opensource.com/sites/default/files/u128651/battle_tanks.png)
|
||||||
|
|
||||||
|
[坦克战役][4] 是一个俯瞰式视角的快节奏坦克战斗游戏。玩家可以选择三种不同的陆地坦克,操纵其在地图上前行,收集道具并且尝试炸飞敌军。它有四种游戏模式,死亡竞赛(又称“死斗”)、团队死斗、夺旗模式和合作模式。死斗和夺旗模式下,分别有 9 张地图可供玩家选择,合作模式则有 4 张。该游戏支持分屏本地双人游戏,以及在线多人竞技。游戏节奏很快,默认一次战役仅 5 分钟,因此,坦克战役十分适合想要利用零碎时间快速来一局的玩家。
|
||||||
|
|
||||||
|
安装坦克战役,你只需要在终端下运行以下指令:
|
||||||
|
|
||||||
|
- Fedora 用户: `dnf install btanks`
|
||||||
|
- Debian/Ubuntu 用户: `apt install btanks`
|
||||||
|
|
||||||
|
### <ruby>火星<rt>M.A.R.S.</rt></ruby>
|
||||||
|
|
||||||
|
![](https://opensource.com/sites/default/files/u128651/m.a.r.s.png)
|
||||||
|
|
||||||
|
[火星][5] 是一个自上而下的太空射击游戏,游戏机制类似传统街机游戏 “<ruby>爆破彗星<rt>Asteroids</rt></ruby>”。玩家在操控一个太空船的同时向敌方射击并躲避敌军的弹幕射击。游戏有标准的死斗和团体死斗模式,除此之外也有更新鲜的比赛形式 —— 例如在一个模式下,玩家需要控制一个球使其进入敌方母星。该游戏支持本地多人游戏,但遗憾的是不支持多人联机。该游戏的开发更新似乎已经停止,所以该游戏之后增加联机模式的几率很小,但就算没有联机支持,这个游戏仍然值得一试。
|
||||||
|
|
||||||
|
安装火星,你只需要在终端下运行以下指令:
|
||||||
|
|
||||||
|
- Fedora 用户: `dnf install marsshooter`
|
||||||
|
- Debian/Ubuntu 用户: `apt install marsshooter`
|
||||||
|
|
||||||
|
### <ruby>不存在之球<rt>Neverball</rt></ruby>
|
||||||
|
|
||||||
|
![](https://opensource.com/sites/default/files/u128651/neverball.png)
|
||||||
|
|
||||||
|
[不存在之球][6] 的游戏灵感来源自世嘉的 “<ruby>超级猴子球<rt>Super Monkey Ball</rt></ruby>” ,玩家需要将一个球在 3D 球场上运动起来,但是玩家控制的不是球,而是球场。游戏任务是在规定的时限内,收集足够多的金币从而打开该关卡的出口并且将小球落进该洞中。游戏可以调整难度,从休闲到难以超乎想象,可以适应不同的玩家需求。该游戏支持键盘/鼠标以及控制杆操作。
|
||||||
|
|
||||||
|
安装不存在之球,你只需要在终端下运行以下指令:
|
||||||
|
|
||||||
|
- Fedora 用户:`dnf install neverball`
|
||||||
|
- Debian/Ubuntu 用户:`apt install neverball`
|
||||||
|
|
||||||
|
### <ruby>超级 Tux<rt>SuperTux</rt></ruby>
|
||||||
|
|
||||||
|
![](https://opensource.com/sites/default/files/u128651/supertux.png)
|
||||||
|
|
||||||
|
[超级 Tux][7] 是继任天堂超级马里奥后的一款 2D 的平台跳跃游戏。Linux 的吉祥物企鹅 Tux 代替了马里奥,而鸡蛋对应着马里奥系列中的蘑菇能力提升。当 Tux 获得了鸡蛋得到了能力提升,它便可以收集花朵,而花朵可以带来新的附加特殊能力。火焰花在关卡中最为常见,收集了火焰花的 Tux 可以掷出火球。除此之外,冰冻花/空气花/土地花也在游戏的程序中。收集星星的能力提升能使 Tux 暂时变得隐形,就如同马里奥系列游戏。该游戏最基础的一组关卡,冰之岛也有 30 关之多,因此游戏的内容和流程和超级马里奥系列一般长。SuperTux 还有一些附加关卡,例如三个额外奖励小岛、一个森林之岛、一个万圣节岛、一个孵化处,以及很多测试关卡。SuperTux 有一个自带的关卡编辑器,所以玩家可以创建他们的原创关卡。
|
||||||
|
|
||||||
|
安装超级 Tux,你只需要在终端下运行以下指令:
|
||||||
|
|
||||||
|
- Fedora 用户:`dnf install supertux`
|
||||||
|
- Debian/Ubuntu 用户: `apt install supertux`
|
||||||
|
|
||||||
|
如果我没有在上文中提及你最喜欢的开源街机游戏,欢迎在评论中分享。
|
||||||
|
|
||||||
|
### 有关作者
|
||||||
|
|
||||||
|
Joshua Allen Holm - 是 Opensource.com 的社区协调者之一。他的主要兴趣有数字人文、学术开放以及公开教育资源。你可以在 GitHub、GitLab、LinkedIn 和 Zotero 上找到他。可以通过 holmja@opensource.com 联系到他。
|
||||||
|
|
||||||
|
------
|
||||||
|
|
||||||
|
via: https://opensource.com/article/18/1/arcade-games-linux
|
||||||
|
|
||||||
|
作者:[Joshua Allen Holm][a]
|
||||||
|
译者:[Scoutydren](https://github.com/Scoutydren)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]: https://opensource.com/users/holmja
|
||||||
|
[1]: http://store.steampowered.com/
|
||||||
|
[2]: https://www.gog.com/
|
||||||
|
[3]: http://www.viewizard.com/
|
||||||
|
[4]: http://btanks.sourceforge.net/blog/about-game
|
||||||
|
[5]: http://mars-game.sourceforge.net/?page_id=10
|
||||||
|
[6]: https://neverball.org/index.php
|
||||||
|
[7]: http://supertux.org/
|
@ -0,0 +1,144 @@
|
|||||||
|
如何使用 Ansible 管理你的工作站配置
|
||||||
|
======
|
||||||
|
|
||||||
|
> 在这个系列的第一篇中,学习一下管理笔记本电脑和台式机配置的基础内容。
|
||||||
|
|
||||||
|
![](https://opensource.com/sites/default/files/styles/image-full-size/public/lead-images/computer_keyboard_laptop_development_code_woman.png?itok=vbYz6jjb)
|
||||||
|
|
||||||
|
配置管理是服务器管理和 DevOps 的一个非常重要的方面。“<ruby>基础架构即代码<rt>infrastructure as code</rt></ruby>”方法可以轻松地以各种配置部署服务器,并动态扩展组织的资源以满足用户需求。但是,对于希望自动设置自己的笔记本电脑和台式机(工作站)的个人管理员的关注较少。
|
||||||
|
|
||||||
|
在本系列中,我将向你展示如何通过 [Ansible][1] 自动化你的工作站设置,如果你想要或需要重新安装你的机器,这可以让你轻松恢复整个配置。此外,如果你有多个工作站,则可以使用相同的方法在每个工作站上进行相同的配置。在第一篇文章中,我们将为个人或工作计算机设置基本的配置管理,并为本系列的其余部分奠定基础。到本文结束时,你将会因此得到一个可以工作的环境。本系列之后的每篇文章都会自动化更多内容并增加复杂性。
|
||||||
|
|
||||||
|
### 为什么用 Ansible?
|
||||||
|
|
||||||
|
有许多配置管理解决方案,包括 Salt Stack、Chef 和 Puppet。我更喜欢 Ansible,因为它在资源利用方面更轻量级,语法更容易阅读,并且如果正确使用它可以彻底改变你的配置管理。Ansible 的轻量级特性与这个主题特别相关,因为我们可能不希望运行一整台服务器而只是为了自动化我们的笔记本电脑和台式机的设置。一般我们总是想要快一些;我们可以使用某些东西来快速启动和运行,以在我们需要恢复的工作站或在多台机器之间同步我们的配置。我使用 Ansible 的具体方法(我将在本文中演示)非常适用于此,而不需要维护服务器。你只需下载配置并运行它。
|
||||||
|
|
||||||
|
### 我的方法
|
||||||
|
|
||||||
|
通常,Ansible 运行于中央服务器。它使用一个<ruby>库存清单<rt>inventory</rt></ruby>文件,该文件是一个文本文件,其中包含我们希望 Ansible 管理的所有主机及其 IP 地址或域名的列表。这对于静态环境非常有用,但对于工作站来说并不理想。原因是我们真的不知道我们的工作站在某一时刻的状态。也许我关闭了台式电脑,或者笔记本电脑可能会被挂起并放在我的包里。在任何一种情况下,Ansible 服务器都会抱怨,因为如果它们处于脱机状态,Ansible 就无法联系到我的机器。我们更需要的是按需方式,我们通过利用 `ansible-pull` 来实现这一目标。`ansible-pull` 命令是 Ansible 的一个命令,允许你从 Git 仓库下载配置并立即应用它。你不需要维护服务器或库存清单;你只需运行 `ansible-pull` 命令,给它一个 Git 仓库 URL,它将为你完成剩下的工作。
|
||||||
|
|
||||||
|
### 起步
|
||||||
|
|
||||||
|
首先,在要管理的计算机上安装 Ansible。有一个问题是许多发行版都附带了旧版本的 Ansible。根据经验,你肯定希望获得最新版本。Ansible 中经常引入新功能,如果你运行的是旧版本,则你在网上找到的示例语法可能无法正常运行,因为它使用的功能未在你安装的版本中实现。甚至发布的小版本都有很多新功能。其中一个例子是 `dconf` 模块,它是从 Ansible 2.4 开始的新功能。如果你尝试使用使用此模块的语法,除非你使用 2.4 或更新版本,否则会失败。在 Ubuntu 及其衍生产品中,我们可以使用官方个人包存档([PPA][2])轻松安装最新版本的 Ansible。以下命令可以解决这个问题:
|
||||||
|
|
||||||
|
```
|
||||||
|
sudo apt-get install software-properties-common
|
||||||
|
sudo apt-add-repository ppa:ansible/ansible
|
||||||
|
sudo apt-get update
|
||||||
|
sudo apt-get install ansible
|
||||||
|
```
|
||||||
|
|
||||||
|
如果你没有使用 Ubuntu,请参阅 [Ansible 的文档][3] 了解如何为你的平台获取它。
|
||||||
|
|
||||||
|
接下来,我们需要一个 Git 仓库来保存我们的配置。满足此要求的最简单方法是在 GitHub 上创建一个空的仓库,或者如果有的话,也可以使用自己的 Git 服务器。为了简单起见,我假设你正在使用 GitHub,因此如果你正在使用其他仓库,请相应调整命令。在 GitHub 中创建一个仓库;你最终会得到一个与此类似的仓库 URL:
|
||||||
|
|
||||||
|
```
|
||||||
|
git@github.com:<your_user_name>/ansible.git
|
||||||
|
```
|
||||||
|
|
||||||
|
将该仓库克隆到你的本地工作目录(忽略任何抱怨仓库为空的消息):
|
||||||
|
|
||||||
|
```
|
||||||
|
git clone git@github.com:<your_user_name>/ansible.git
|
||||||
|
```
|
||||||
|
|
||||||
|
现在我们有了一个可以使用的空仓库。将你的工作目录切换到仓库(例如 `cd ./ansible`),并在你喜欢的文本编辑器中创建名为 `local.yml` 的文件。将以下配置放在该文件中:
|
||||||
|
|
||||||
|
```
|
||||||
|
- hosts: localhost
|
||||||
|
become: true
|
||||||
|
tasks:
|
||||||
|
- name: Install htop
|
||||||
|
apt: name=htop
|
||||||
|
```
|
||||||
|
|
||||||
|
你刚刚创建的文件被称为<ruby>剧本<rt>playbook</rt></ruby>,安装 `htop` 的指令(我任意选择的一个包作为例子)被称为<ruby>动作<rt>play</rt></ruby>。剧本本身是一个 YAML 格式的文件,它是一种易于阅读的标记语言。对 YAML 的完整讲述超出了本文的范围,但你无需专业理解即可熟练使用 Ansible。该配置易于阅读;只需查看此文件,你就可以轻松理解我们正在安装的 `htop` 软件包。要注意一下最后一行的 `apt` 模块,它只适用于基于 Debian 的系统。如果你使用的是 Red Hat 平台,你可以将其更改为 `yum` 而不是 `apt`,或者如果你正在使用 Fedora,则将其更改为 `dnf`。`name` 行只是提供有关我们任务的信息,并将显示在输出中。因此,你需要确保名称具有描述性,以便在需要对多个动作进行故障排除时很容易找到。
|
||||||
|
|
||||||
|
接下来,让我们将新文件提交到我们的仓库:
|
||||||
|
|
||||||
|
```
|
||||||
|
git add local.yml
|
||||||
|
git commit -m "initial commit"
|
||||||
|
git push origin master
|
||||||
|
```
|
||||||
|
|
||||||
|
现在我们的新剧本应该出现在我们的 GitHub 上的仓库中。我们可以使用以下命令应用我们创建的剧本:
|
||||||
|
|
||||||
|
```
|
||||||
|
sudo ansible-pull -U https://github.com/<your_user_name>/ansible.git
|
||||||
|
```
|
||||||
|
|
||||||
|
如果执行正确,`htop`包应该会安装在你的系统上。你可能会在开头附近看到一些警告,抱怨缺少库存清单文件。这很好,因为我们没有使用库存清单文件(我们也不需要这样做)。在输出结束时,它将概述它做的内容。如果 `htop` 安装正确,你应该在输出的最后一行看到 `changed = 1`。
|
||||||
|
|
||||||
|
它是如何工作的呢?`ansible-pull` 命令使用了 `-U` 选项,它需要一个仓库 URL。出于安全考虑,我给它提供了仓库 URL 的 https 版本,因为我不希望任何主机对仓库具有写访问权限(默认情况下 https 是只读的)。`local.yml` 是预设的剧本名称,因此我们不需要为剧本提供文件名:如果它在仓库的根目录中找到名为 `local.yml` 的剧本,它将自动运行它。接下来,我们在命令前面使用了 `sudo`,因为我们正在修改系统。
|
||||||
|
|
||||||
|
让我们继续为我们的剧本添加更多的包。我将添加两个包,使它看起来像这样:
|
||||||
|
|
||||||
|
```
|
||||||
|
- hosts: localhost
|
||||||
|
become: true
|
||||||
|
tasks:
|
||||||
|
- name: Install htop
|
||||||
|
apt: name=htop
|
||||||
|
|
||||||
|
- name: Install mc
|
||||||
|
apt: name=mc
|
||||||
|
|
||||||
|
- name: Install tmux
|
||||||
|
apt: name=tmux
|
||||||
|
```
|
||||||
|
|
||||||
|
我添加了更多的动作(任务)来安装另外两个包,`mc` 和 `tmux`。在此剧本中选择安装的哪些软件包并不重要;我只是随意挑选这些。你应该安装你希望所有的系统都具有的软件包。唯一需要注意的是,在你分发前,你必须知道那个包存在于软件仓库中。
|
||||||
|
|
||||||
|
在我们提交并应用这个更新的剧本之前,我们应该整理一下它。它可以很好地工作,但(说实话)它看起来有点混乱。让我们尝试在一个动作中安装所有三个包。用下面这个替换你的 `local.yml` 的内容:
|
||||||
|
|
||||||
|
```
|
||||||
|
- hosts: localhost
|
||||||
|
become: true
|
||||||
|
tasks:
|
||||||
|
- name: Install packages
|
||||||
|
apt: name={{item}}
|
||||||
|
with_items:
|
||||||
|
- htop
|
||||||
|
- mc
|
||||||
|
- tmux
|
||||||
|
```
|
||||||
|
|
||||||
|
现在看起来更干净、更有效率了。我们使用 `with_items` 将我们的包列表合并为一个动作。如果我们想要添加其他包,我们只需添加另一个带有连字符和包名称的行。可以把 `with_items` 看做类似于 `for` 循环。我们列出的每个包都将安装。
|
||||||
|
|
||||||
|
将我们的新更改提交回仓库:
|
||||||
|
|
||||||
|
```
|
||||||
|
git add local.yml
|
||||||
|
git commit -m "added additional packages, cleaned up formatting"
|
||||||
|
git push origin master
|
||||||
|
```
|
||||||
|
|
||||||
|
现在我们可以运行我们的剧本以接受新的新配置:
|
||||||
|
|
||||||
|
```
|
||||||
|
sudo ansible-pull -U https://github.com/<your_user_name>/ansible.git
|
||||||
|
```
|
||||||
|
|
||||||
|
不可否认,这个例子还没有做多少事情;它所做的就是安装一些软件包。你可以使用包管理器更快地安装这些包。然而,随着这个系列的继续,这些例子将变得更加复杂,我们将自动化更多的东西。最后,你创建的 Ansible 配置将自动执行越来越多的任务。例如,我自己使用的那个配置可以自动安装数百个软件包、设置cron 作业、处理桌面配置等等。
|
||||||
|
|
||||||
|
从我们迄今为止所取得的成就来看,你可能已经有了大概了解。我们所要做的就是创建一个仓库,在该仓库中放置一个剧本,然后利用 `ansible-pull` 命令拉取该仓库并将其应用到我们的机器上。我们不需要设置服务器。将来,如果我们想要更改配置,我们可以拉取该仓库、更新它,然后将其推回到我们的仓库并应用它。如果我们要设置新机器,我们只需要安装 Ansible 并应用配置。
|
||||||
|
|
||||||
|
在下一篇文章中,我们将通过 cron 和一些其他项目进一步自动化。与此同时,我已将本文的代码复制到
|
||||||
|
[我的 GitHub 仓库][4] 中,以便你可以用你的语法对比一下我的。随着我们的进展,我会不断更新代码。
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: https://opensource.com/article/18/3/manage-workstation-ansible
|
||||||
|
|
||||||
|
作者:[Jay LaCroix][a]
|
||||||
|
译者:[wxy](https://github.com/)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]:https://opensource.com/users/jlacroix
|
||||||
|
[1]:https://www.ansible.com/
|
||||||
|
[2]:https://launchpad.net/ubuntu/+ppas
|
||||||
|
[3]:http://docs.ansible.com/ansible/latest/intro_installation.html
|
||||||
|
[4]:https://github.com/jlacroix82/ansible_article
|
@ -0,0 +1,154 @@
|
|||||||
|
用 PGP 保护代码完整性(四):将主密钥移到离线存储中
|
||||||
|
======
|
||||||
|
> 如果开发者的 PGP 密钥被偷了,危害非常大。了解一下如何更安全。
|
||||||
|
|
||||||
|
![](https://www.linux.com/sites/lcom/files/styles/rendered_file/public/industry-1920.jpg?itok=gI3QraS8)
|
||||||
|
|
||||||
|
在本系列教程中,我们为使用 PGP 提供了一个实用指南。你可以从下面的链接中查看前面的文章:
|
||||||
|
|
||||||
|
- [第一部分:基本概念和工具][1]
|
||||||
|
- [第二部分:生成你的主密钥][2]
|
||||||
|
- [第三部分:生成 PGP 子密钥][3]
|
||||||
|
|
||||||
|
这是本系列教程的第四部分,我们继续本教程,我们将谈一谈如何及为什么要将主密钥从你的家目录移到离线存储中。现在开始我们的教程。
|
||||||
|
|
||||||
|
#### 清单
|
||||||
|
|
||||||
|
* 准备一个加密的可移除的存储(必要)
|
||||||
|
* 备份你的 GnuPG 目录(必要)
|
||||||
|
* 从你的家目录中删除主密钥(推荐)
|
||||||
|
* 从你的家目录中删除吊销证书(推荐)
|
||||||
|
|
||||||
|
#### 考虑事项
|
||||||
|
|
||||||
|
为什么要从你的家目录中删除你的主 [C] 密钥 ?这样做的主要原因是防止你的主密钥失窃或意外泄露。对于心怀不轨的人来说,私钥对他们具有很大的诱惑力 —— 我们知道有几个恶意软件成功地实现了扫描用户的家目录并将发现的私钥内容上传。
|
||||||
|
|
||||||
|
对于开发者来说,私钥失窃是非常危险的事情 —— 在自由软件的世界中,这无疑是身份证明失窃。从你的家目录中删除私钥将帮你防范这类事件的发生。
|
||||||
|
|
||||||
|
#### 备份你的 GnuPG 目录
|
||||||
|
|
||||||
|
**!!!绝对不要跳过这一步!!!**
|
||||||
|
|
||||||
|
备份你的 PGP 密钥将让你在需要的时候很容易地恢复它们,这很重要!(这与我们做的使用 paperkey 的灾难级备份是不一样的)。
|
||||||
|
|
||||||
|
#### 准备可移除的加密存储
|
||||||
|
|
||||||
|
我们从取得一个(最好是两个)小型的 USB “拇指“ 驱动器(可加密 U 盘)开始,我们将用它来做备份。你首先需要去加密它们:
|
||||||
|
|
||||||
|
加密密码可以使用与主密钥相同的密码。
|
||||||
|
|
||||||
|
#### 备份你的 GnuPG 目录
|
||||||
|
|
||||||
|
加密过程结束之后,重新插入 USB 驱动器并确保它能够正常挂载。你可以通过运行 `mount` 命令去找到设备挂载点的完全路径。(在 Linux 下,外置介质一般挂载在 `/media/disk` 下,Mac 一般在它的 `/Volumes` 下)
|
||||||
|
|
||||||
|
你知道了挂载点的全路径后,将你的整个 GnuPG 的目录复制进去:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ cp -rp ~/.gnupg [/media/disk/name]/gnupg-backup
|
||||||
|
```
|
||||||
|
|
||||||
|
(注意:如果出现任何套接字不支持的错误,没有关系,直接忽略它们。)
|
||||||
|
|
||||||
|
现在,用如下的命令去测试一下,确保它们能够正常地工作:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ gpg --homedir=[/media/disk/name]/gnupg-backup --list-key [fpr]
|
||||||
|
```
|
||||||
|
|
||||||
|
如果没有出现任何错误,说明一切正常。弹出这个 USB 驱动器并给它粘上一个明确的标签,以便于你下次需要它时能够很快找到它。接着,将它放到一个安全的 —— 但不要太远 —— 的地方,因为从现在起,你需要偶尔使用它来做一些像编辑身份信息、添加或吊销子证书、或签署其它人的密钥这样的事情。
|
||||||
|
|
||||||
|
#### 删除主密钥
|
||||||
|
|
||||||
|
我们家目录中的文件并没有像我们所想像的那样受到保护。它们可能会通过许多不同的方式被泄露或失窃:
|
||||||
|
|
||||||
|
* 通过快速复制来配置一个新工作站时的偶尔事故
|
||||||
|
* 通过系统管理员的疏忽或恶意操作
|
||||||
|
* 通过安全性欠佳的备份
|
||||||
|
* 通过桌面应用中的恶意软件(浏览器、pdf 查看器等等)
|
||||||
|
* 通过跨境胁迫
|
||||||
|
|
||||||
|
使用一个很好的密码来保护你的密钥是降低上述风险的一个很好方法,但是密码能够通过键盘记录器、背后窥视、或其它方式被发现。基于以上原因,我们建议去配置一个从你的家目录上可移除的主密钥,将它保存在一个离线存储中。
|
||||||
|
|
||||||
|
##### 删除你的主密钥
|
||||||
|
|
||||||
|
**请查看前面的节,确保你有完整的你的 GnuPG 目录的一个备份。如果你没有一个可用的备份,下面所做的操作将会使你的主密钥失效!!!**
|
||||||
|
|
||||||
|
首先,识别你的主密钥的 keygrip:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ gpg --with-keygrip --list-key [fpr]
|
||||||
|
```
|
||||||
|
|
||||||
|
它的输出应该像下面这样:
|
||||||
|
|
||||||
|
```
|
||||||
|
pub rsa4096 2017-12-06 [C] [expires: 2019-12-06]
|
||||||
|
111122223333444455556666AAAABBBBCCCCDDDD
|
||||||
|
Keygrip = AAAA999988887777666655554444333322221111
|
||||||
|
uid [ultimate] Alice Engineer <alice@example.org>
|
||||||
|
uid [ultimate] Alice Engineer <allie@example.net>
|
||||||
|
sub rsa2048 2017-12-06 [E]
|
||||||
|
Keygrip = BBBB999988887777666655554444333322221111
|
||||||
|
sub rsa2048 2017-12-06 [S]
|
||||||
|
Keygrip = CCCC999988887777666655554444333322221111
|
||||||
|
```
|
||||||
|
|
||||||
|
找到 `pub` 行下方的 `Keygrip` 条目(就在主密钥指纹的下方)。它与你的家目录下 `.gnupg` 目录下的一个文件是一致的:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ cd ~/.gnupg/private-keys-v1.d
|
||||||
|
$ ls
|
||||||
|
AAAA999988887777666655554444333322221111.key
|
||||||
|
BBBB999988887777666655554444333322221111.key
|
||||||
|
CCCC999988887777666655554444333322221111.key
|
||||||
|
```
|
||||||
|
|
||||||
|
现在你做的全部操作就是简单地删除与主密钥 keygrip 一致的 `.key` 文件:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ cd ~/.gnupg/private-keys-v1.d
|
||||||
|
$ rm AAAA999988887777666655554444333322221111.key
|
||||||
|
```
|
||||||
|
|
||||||
|
现在,如果运行 `--list-secret-keys` 命令将出现问题,它将显示主密钥丢失(`#` 表示不可用):
|
||||||
|
|
||||||
|
```
|
||||||
|
$ gpg --list-secret-keys
|
||||||
|
sec# rsa4096 2017-12-06 [C] [expires: 2019-12-06]
|
||||||
|
111122223333444455556666AAAABBBBCCCCDDDD
|
||||||
|
uid [ultimate] Alice Engineer <alice@example.org>
|
||||||
|
uid [ultimate] Alice Engineer <allie@example.net>
|
||||||
|
ssb rsa2048 2017-12-06 [E]
|
||||||
|
ssb rsa2048 2017-12-06 [S]
|
||||||
|
```
|
||||||
|
|
||||||
|
#### 删除吊销证书
|
||||||
|
|
||||||
|
你应该去删除的另一个文件是吊销证书(**删除之前,确保你的备份中有它**),它是使用你的主密钥自动创建的。吊销证书允许一些人去永久标记你的证书为吊销状态,这意味着它无论在任何用途中将不再被使用或信任。一般是使用它来吊销由于某些原因不再受控的一个密钥 —— 比如,你丢失了密钥密码。
|
||||||
|
|
||||||
|
与使用主密钥一样,如果一个吊销证书泄露到恶意者手中,他们能够使用它去破坏你的开发者数字身份,因此,最好是从你的家目录中删除它。
|
||||||
|
|
||||||
|
```
|
||||||
|
cd ~/.gnupg/openpgp-revocs.d
|
||||||
|
rm [fpr].rev
|
||||||
|
```
|
||||||
|
|
||||||
|
在下一篇文章中,你将学习如何保护你的子密钥。敬请期待。
|
||||||
|
|
||||||
|
从来自 Linux 基金会和 edX 的免费课程 [“Linux 入门”][4] 中学习更多 Linux 知识。
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: https://www.linux.com/blog/learn/pgp/2018/3/protecting-code-integrity-pgp-part-4-moving-your-master-key-offline-storage
|
||||||
|
|
||||||
|
作者:[Konstantin Ryabitsev][a]
|
||||||
|
译者:[qhwdw](https://github.com/qhwdw)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]:https://www.linux.com/users/mricon
|
||||||
|
[1]:https://linux.cn/article-9524-1.html
|
||||||
|
[2]:https://linux.cn/article-9529-1.html
|
||||||
|
[3]:https://linux.cn/article-9607-1.html
|
||||||
|
[4]:https://training.linuxfoundation.org/linux-courses/system-administration-training/introduction-to-linux
|
@ -0,0 +1,285 @@
|
|||||||
|
用 PGP 保护代码完整性(五):将子密钥移到一个硬件设备中
|
||||||
|
======
|
||||||
|
|
||||||
|
> 在这个系列教材中,将为你提供使用 PGP 和保护你的私钥的最佳体验。
|
||||||
|
|
||||||
|
![](https://www.linux.com/sites/lcom/files/styles/rendered_file/public/pgp-keys.jpg?itok=aS6IWGpq)
|
||||||
|
|
||||||
|
在本系列教程中,我们将提供一个使用 PGP 的实用指南。如果你没有看过前面的文章,你可以通过下面的链接去查看。在这篇文章中,我们将继续讨论如何保护你的密钥,谈一谈将你的子密钥移到一个专门的硬件设备中的一些技巧。
|
||||||
|
|
||||||
|
- [第一部分:基本概念和工具][1]
|
||||||
|
- [第二部分:生成你的主密钥][2]
|
||||||
|
- [第三部分:生成 PGP 子密钥][3]
|
||||||
|
- [第四部分:将主密钥移到离线存储中][4]
|
||||||
|
|
||||||
|
#### 清单
|
||||||
|
|
||||||
|
* 取得一个 GnuPG 兼容的硬件设备(必要)
|
||||||
|
* 配置 GnuPG 在设备上工作(必要)
|
||||||
|
* 设置用户和管理员的 PIN(必要)
|
||||||
|
* 移动子密钥到设备中(必要)
|
||||||
|
|
||||||
|
#### 考虑事项
|
||||||
|
|
||||||
|
虽然现在主密钥已经不用担心泄露或失窃了,但子密钥仍然在你的家目录中。任何得到它的人都能够解密你的通讯或假冒你的签名(如果他们知道密钥的密码)。并且,每次执行一个 GnuPG 操作都要将密钥加载到操作系统内存中,这将使一些更高级的恶意软件有机会得到你的密钥(想想 Meltdown 和 Spectre)。
|
||||||
|
|
||||||
|
完全保护密钥的最好方式就是,将它移到一个专门的硬件设备中,这种硬件设备是一个可操作的智能卡。
|
||||||
|
|
||||||
|
##### 智能卡的好处
|
||||||
|
|
||||||
|
一个智能卡包含一个加密芯片,它能够存储私钥,并且直接在智能卡内部执行秘密操作。因为密钥内容从来没有离开过智能卡,计算机操作系统并不能检索你插入的智能卡上的私钥。这与前面用于备份目的的加密 USB 存储是不同的 —— 虽然 USB 设备也是插入并解密的,但操作系统是能够去访问私钥内容的。使用外置的加密 USB 介质并不能代替智能卡设备的功能。
|
||||||
|
|
||||||
|
智能卡的一些其它好处:
|
||||||
|
|
||||||
|
* 它们很便宜且易于获得
|
||||||
|
* 它们小巧且易于携带
|
||||||
|
* 它们可以用于多种设备上
|
||||||
|
* 它们中的很多都具有防篡改功能(取决于制造商)
|
||||||
|
|
||||||
|
#### 可用的智能卡设备
|
||||||
|
|
||||||
|
智能卡最初是嵌入到真实钱包大小的卡中,故而得名智能卡。你肯定可以买到并使用 GnuPG 功能的智能卡,并且它们是你能得到的最便宜的可用设备之一。但是,事实上智能卡有一个很重要的缺点:它们需要一个智能卡读卡器,只有极小数的笔记本电脑上有这种读卡器。
|
||||||
|
|
||||||
|
由于这个原因,制造商开始推出小型 USB 设备,它的大小和 U 盘类似,内置有微型智能卡,并且在芯片上简单地实现了智能卡协议特性。下面推荐几个这样的设备:
|
||||||
|
|
||||||
|
* [Nitrokey Start][5]:开源硬件和自由软件,可用于 GnuPG 的最便宜的选择之一,但是额外的安全特性很少。
|
||||||
|
* [Nitrokey Pro][6]:类似于 Nitrokey Start,它提供防篡改及更多的安全特性(但没有 U2F,具体查看指南的 U2F 节)。
|
||||||
|
* [Yubikey 4][7]:专利硬件和软件,但比 Nitrokey Pro 便宜,并且可以用在最新的笔记本电脑上的 USB-C 接口;也提供像 U2F 这样的额外的安全特性。
|
||||||
|
|
||||||
|
我们推荐选一个同时具备智能卡功能和 U2F 的设备,在写这篇文章时,只能选择 Yubikey 4。
|
||||||
|
|
||||||
|
#### 配置智能卡设备
|
||||||
|
|
||||||
|
你的智能卡设备插入任何一台现代的 Linux 或 Mac 工作站上都应该能正常工作。你可以通过运行如下的命令去验证它:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ gpg --card-status
|
||||||
|
```
|
||||||
|
|
||||||
|
如果你没有收到错误,有一个完整的卡列表,就表示一切正常。不幸的是,排除为什么设备不能正常工作的所有可能原因,已经超出了本指南的范围。如果你的智能卡使用 GnuPG 时有问题,请通过你的操作系统的常见支持通道寻求支持。
|
||||||
|
|
||||||
|
##### PIN 不一定是数字
|
||||||
|
|
||||||
|
注意,尽管名为 “PIN”(暗示你它必须是一个“数字”),不论是用户 PIN 还是管理员 PIN 都不必非要是数字。
|
||||||
|
|
||||||
|
当你收到一个新设备时,它可能设置有一个默认的用户和管理员 PIN,对于 Yubikey,它分别是 `123456` 和 `12345678`。如果它们的 PIN 不是默认的,请查看设备附带的说明书。
|
||||||
|
|
||||||
|
##### 快速设置
|
||||||
|
|
||||||
|
为配置你的智能卡,你需要使用 GnuPG 菜单系统,因此这里并没有更方便的命令行开关:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ gpg --card-edit
|
||||||
|
[...omitted...]
|
||||||
|
gpg/card> admin
|
||||||
|
Admin commands are allowed
|
||||||
|
gpg/card> passwd
|
||||||
|
```
|
||||||
|
|
||||||
|
你应该去设置用户 PIN (1)、管理员 PIN (3)、和重置码 (4)。请确保把它们记录并保存到一个安全的地方 —— 尤其是管理员 PIN 和重置码(它允许你去擦除整个智能卡内容)。你很少使用到管理员 PIN,因此如果你不记录下来,很可能会忘掉它。
|
||||||
|
|
||||||
|
返回到智能卡主菜单,你也可以设置其它值(比如名字、性别、登入日期、等等),但是这些都不是必需的,一旦你的智能卡丢失了,将导致额外的信息泄露。
|
||||||
|
|
||||||
|
#### 将子密钥移到你的智能卡中
|
||||||
|
|
||||||
|
退出卡菜单(使用 `q` 命令)保存所有更改。接下来,我们将你的子密钥移到智能卡中。将需要用到你的 PGP 密钥的密码,在大多数的智能卡操作中都将用到管理员 PIN。记住,那个 `[fpr]` 表示你的密钥的完整的 40 个字符的指纹。
|
||||||
|
|
||||||
|
```
|
||||||
|
$ gpg --edit-key [fpr]
|
||||||
|
|
||||||
|
Secret subkeys are available.
|
||||||
|
|
||||||
|
pub rsa4096/AAAABBBBCCCCDDDD
|
||||||
|
created: 2017-12-07 expires: 2019-12-07 usage: C
|
||||||
|
trust: ultimate validity: ultimate
|
||||||
|
ssb rsa2048/1111222233334444
|
||||||
|
created: 2017-12-07 expires: never usage: E
|
||||||
|
ssb rsa2048/5555666677778888
|
||||||
|
created: 2017-12-07 expires: never usage: S
|
||||||
|
[ultimate] (1). Alice Engineer <alice@example.org>
|
||||||
|
[ultimate] (2) Alice Engineer <allie@example.net>
|
||||||
|
|
||||||
|
gpg>
|
||||||
|
```
|
||||||
|
|
||||||
|
使用 `--edit-key` 再次进入到菜单模式,你将注意到那个密钥清单有一点小差别。从现在开始,所有的命令都是在这个菜单模式下运行,它用 `gpg>` 提示符来表示。
|
||||||
|
|
||||||
|
首先,我们来选择移到智能卡中的密钥 —— 你可以通过键入 `key 1`(它表示选择清单中的第一个密钥)来实现:
|
||||||
|
|
||||||
|
```
|
||||||
|
gpg> key 1
|
||||||
|
```
|
||||||
|
|
||||||
|
这个输出会有一点细微的差别:
|
||||||
|
|
||||||
|
```
|
||||||
|
pub rsa4096/AAAABBBBCCCCDDDD
|
||||||
|
created: 2017-12-07 expires: 2019-12-07 usage: C
|
||||||
|
trust: ultimate validity: ultimate
|
||||||
|
ssb* rsa2048/1111222233334444
|
||||||
|
created: 2017-12-07 expires: never usage: E
|
||||||
|
ssb rsa2048/5555666677778888
|
||||||
|
created: 2017-12-07 expires: never usage: S
|
||||||
|
[ultimate] (1). Alice Engineer <alice@example.org>
|
||||||
|
[ultimate] (2) Alice Engineer <allie@example.net>
|
||||||
|
```
|
||||||
|
|
||||||
|
注意与密钥对应的 `ssb` 行旁边的 `*` —— 它表示这是当前选定的密钥。它是可切换的,意味着如果你再次输入 `key 1`,这个 `*` 将消失,这个密钥将不再被选中。
|
||||||
|
|
||||||
|
现在,我们来将密钥移到智能卡中:
|
||||||
|
|
||||||
|
```
|
||||||
|
gpg> keytocard
|
||||||
|
Please select where to store the key:
|
||||||
|
(2) Encryption key
|
||||||
|
Your selection? 2
|
||||||
|
```
|
||||||
|
|
||||||
|
由于它是我们的 [E] 密钥,把它移到加密区中是有很有意义的。当你提交了你的选择之后,将会被提示输入你的 PGP 密钥的保护密码,接下来输入智能卡的管理员 PIN。如果命令没有返回错误,表示你的密钥已经被移到智能卡中了。
|
||||||
|
|
||||||
|
**重要:** 现在再次输入 `key 1` 去取消选中第一个密钥,并输入 `key 2` 去选择 [S] 密钥:
|
||||||
|
|
||||||
|
```
|
||||||
|
gpg> key 1
|
||||||
|
gpg> key 2
|
||||||
|
gpg> keytocard
|
||||||
|
Please select where to store the key:
|
||||||
|
(1) Signature key
|
||||||
|
(3) Authentication key
|
||||||
|
Your selection? 1
|
||||||
|
```
|
||||||
|
|
||||||
|
你可以使用 [S] 密钥同时做签名和验证,但是我们希望确保它在签名区,因此,我们选择 (`1`)。完成后,如果你的命令没有返回错误,表示操作已成功。
|
||||||
|
|
||||||
|
最后,如果你创建了一个 [A] 密钥,你也可以将它移到智能卡中,但是你需要先取消选中 `key 2`。完成后,选择 `q`:
|
||||||
|
|
||||||
|
```
|
||||||
|
gpg> q
|
||||||
|
Save changes? (y/N) y
|
||||||
|
```
|
||||||
|
|
||||||
|
保存变更将把你的子密钥移到智能卡后,把你的家目录中的相应子密钥删除(没有关系,因为我们的备份中还有,如果更换了智能卡,你需要再做一遍)。
|
||||||
|
|
||||||
|
##### 验证移动后的密钥
|
||||||
|
|
||||||
|
现在,如果你执行一个` --list-secret-keys` 操作,你将看到一个稍有不同的输出:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ gpg --list-secret-keys
|
||||||
|
sec# rsa4096 2017-12-06 [C] [expires: 2019-12-06]
|
||||||
|
111122223333444455556666AAAABBBBCCCCDDDD
|
||||||
|
uid [ultimate] Alice Engineer <alice@example.org>
|
||||||
|
uid [ultimate] Alice Engineer <allie@example.net>
|
||||||
|
ssb> rsa2048 2017-12-06 [E]
|
||||||
|
ssb> rsa2048 2017-12-06 [S]
|
||||||
|
```
|
||||||
|
|
||||||
|
在 `ssb>` 的输出中的 `>` 表示子密钥仅在智能卡上有效。如果你进入到你的密钥目录中,查看目录的内容,你将会看到那个 `.key` 文件已经被存根替换:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ cd ~/.gnupg/private-keys-v1.d
|
||||||
|
$ strings *.key
|
||||||
|
```
|
||||||
|
|
||||||
|
这个输出将包含一个影子私钥,它表示那个文件仅是个存根,真正的内容在智能卡中。
|
||||||
|
|
||||||
|
#### 验证智能卡的功能
|
||||||
|
|
||||||
|
验证智能卡能否如期正常运行,你可以通过创建一个签名来验证:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ echo "Hello world" | gpg --clearsign > /tmp/test.asc
|
||||||
|
$ gpg --verify /tmp/test.asc
|
||||||
|
```
|
||||||
|
|
||||||
|
首次运行这个命令时将询问你智能卡的 PIN,在你运行 `gpg —verify` 之后,它将显示 “Good signature”。
|
||||||
|
|
||||||
|
祝贺你,你已经成功将窃取你的开发者数字身份变得更加困难了!
|
||||||
|
|
||||||
|
#### 其它常见 GnuPG 操作
|
||||||
|
|
||||||
|
下面是使用你的 PGP 密钥需要做的一些常见操作的快速指南。
|
||||||
|
|
||||||
|
在下面的所有命令中,`[fpr]` 表示你的密钥指纹。
|
||||||
|
|
||||||
|
##### 挂载主密钥离线存储
|
||||||
|
|
||||||
|
下面的一些操作将需要你的主密钥,因此首先需要去挂载你的主密钥离线存储,并告诉 GnuPG 去使用它。首先,找出介质挂载路径,可以通过查看 `mount` 命令的输出找到它。接着,设置你的 GnuPG 目录为你的介质上备份的目录,并告诉 GnuPG 将那个目录做为它的家目录:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ export GNUPGHOME=/media/disk/name/gnupg-backup
|
||||||
|
$ gpg --list-secret-keys
|
||||||
|
```
|
||||||
|
|
||||||
|
确保你在输出中看到的是 `sec` 而不是 `sec#`(这个 `#` 表示密钥不可用,仍然使用的是惯常的那个 Home 目录)。
|
||||||
|
|
||||||
|
##### 更新你惯常使用的那个 GnuPG 工作目录
|
||||||
|
|
||||||
|
在你的离线存储上做了任何更改之后,你应该将这些更改同步应用到你惯常使用的工作目录中:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ gpg --export | gpg --homedir ~/.gnupg --import
|
||||||
|
$ unset GNUPGHOME
|
||||||
|
```
|
||||||
|
|
||||||
|
##### 延长密钥过期日期
|
||||||
|
|
||||||
|
我们创建的主密钥的默认过期日期是自创建之日起两年后。这样做都是为安全考虑,这样将使淘汰密钥最终从密钥服务器上消失。
|
||||||
|
|
||||||
|
延长你的密钥过期日期,从当前日期延长一年,只需要运行如下命令:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ gpg --quick-set-expire [fpr] 1y
|
||||||
|
```
|
||||||
|
|
||||||
|
如果为了好记住,你也可以使用一个特定日期(比如,你的生日、1 月 1 日、或加拿大国庆日):
|
||||||
|
|
||||||
|
```
|
||||||
|
$ gpg --quick-set-expire [fpr] 2020-07-01
|
||||||
|
```
|
||||||
|
|
||||||
|
记得将更新后的密钥发送到密钥服务器:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ gpg --send-key [fpr]
|
||||||
|
```
|
||||||
|
|
||||||
|
##### 吊销身份
|
||||||
|
|
||||||
|
如果你需要吊销一个身份(比如,你换了雇主并且旧的邮件地址不再有效了),你可以使用一行命令搞定:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ gpg --quick-revoke-uid [fpr] 'Alice Engineer <aengineer@example.net>'
|
||||||
|
```
|
||||||
|
|
||||||
|
你也可以通过使用 `gpg --edit-key [fpr]` 在菜单模式下完成同样的事情。
|
||||||
|
|
||||||
|
完成后,记得将更新后的密钥发送到密钥服务器上:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ gpg --send-key [fpr]
|
||||||
|
```
|
||||||
|
|
||||||
|
下一篇文章中,我们将谈谈 Git 如何支持 PGP 的多级别集成。
|
||||||
|
|
||||||
|
通过来自 Linux 基金会和 edX 的免费课程 [“Linux 入门”][8]学习更多 Linux 知识。
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: https://www.linux.com/blog/learn/pgp/2018/3/protecting-code-integrity-pgp-part-5-moving-subkeys-hardware-device
|
||||||
|
|
||||||
|
作者:[KONSTANTIN RYABITSEV][a]
|
||||||
|
译者:[qhwdw](https://github.com/qhwdw)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]:https://www.linux.com/users/mricon
|
||||||
|
[1]:https://linux.cn/article-9524-1.html
|
||||||
|
[2]:https://linux.cn/article-9529-1.html
|
||||||
|
[3]:https://linux.cn/article-9607-1.html
|
||||||
|
[4]:https://linux.cn/article-10402-1.html
|
||||||
|
[5]:https://shop.nitrokey.com/shop/product/nitrokey-start-6
|
||||||
|
[6]:https://shop.nitrokey.com/shop/product/nitrokey-pro-3
|
||||||
|
[7]:https://www.yubico.com/product/yubikey-4-series/
|
||||||
|
[8]:https://training.linuxfoundation.org/linux-courses/system-administration-training/introduction-to-linux
|
@ -0,0 +1,283 @@
|
|||||||
|
用 PGP 保护代码完整性(六):在 Git 上使用 PGP
|
||||||
|
======
|
||||||
|
|
||||||
|
> 我们继续我们的 PGP 实践系列,来看看签名标签的标签和提交,这可以帮你确保你的仓库没有被篡改。
|
||||||
|
|
||||||
|
![](https://www.linux.com/sites/lcom/files/styles/rendered_file/public/global-network.jpg?itok=h_hhZc36)
|
||||||
|
|
||||||
|
在本系列教程中,我们提供了一个使用 PGP 的实用指南,包括基本概念和工具、生成和保护你的密钥。如果你错过了前面的文章,你可以查看下面的链接。在这篇文章中,我们谈一谈在 Git 中如何集成 PGP、使用签名的标签,然后介绍签名提交,最后添加签名推送的支持。
|
||||||
|
|
||||||
|
- [第一部分:基本概念和工具][1]
|
||||||
|
- [第二部分:生成你的主密钥][2]
|
||||||
|
- [第三部分:生成 PGP 子密钥][3]
|
||||||
|
- [第四部分:将主密钥移到离线存储中][4]
|
||||||
|
- [第五部分:将子密钥移到硬件设备中][5]
|
||||||
|
|
||||||
|
Git 的核心特性之一就是它的去中心化本质 —— 一旦仓库克隆到你的本地系统,你就拥有了项目的完整历史,包括所有的标签、提交和分支。然而由于存在着成百上千的克隆仓库,如何才能验证你下载的仓库没有被恶意的第三方做过篡改?你可以从 GitHub 或一些貌似官方的位置来克隆它们,但是如果有些人故意欺骗了你怎么办?
|
||||||
|
|
||||||
|
或者在你参与的一些项目上发现了后门,而 “Author” 行显示是你干的,然而你很确定 [不是你干的][6],会发生什么情况?
|
||||||
|
|
||||||
|
为解决上述问题,Git 添加了 PGP 集成。签名的标签通过确认它的内容与创建这个标签的开发者的工作站上的内容完全一致来证明仓库的完整性,而签名的提交几乎是不可能在不访问你的 PGP 密钥的情况下能够假冒你。
|
||||||
|
|
||||||
|
#### 清单
|
||||||
|
|
||||||
|
* 了解签名的标签、提交和推送(必要)
|
||||||
|
* 配置 git 使用你的密钥(必要)
|
||||||
|
* 学习标签如何签名和验证(必要)
|
||||||
|
* 配置 git 总是签名带注释标签(推荐)
|
||||||
|
* 学习提交如何签名和验证工作(必要)
|
||||||
|
* 配置 git 总是签名提交(推荐)
|
||||||
|
* 配置 gpg-agent 选项(必要)
|
||||||
|
|
||||||
|
#### 考虑事项
|
||||||
|
|
||||||
|
git 实现了 PGP 的多级集成,首先从签名标签开始,接着介绍签名提交,最后添加签名推送的支持。
|
||||||
|
|
||||||
|
##### 了解 Git 哈希
|
||||||
|
|
||||||
|
git 是一个复杂的东西,为了你能够更好地掌握它如何集成 PGP,你需要了解什么是”哈希“。我们将它归纳为两种类型的哈希:树哈希和提交哈希。
|
||||||
|
|
||||||
|
###### 树哈希
|
||||||
|
|
||||||
|
每次你向仓库提交一个变更,对于仓库中的每个子目录,git 都会记录它里面所有对象的校验和哈希 —— 内容(blobs)、目录(trees)、文件名和许可等等。它只对每次提交中发生变更的树和内容做此操作,这样在只变更树的一小部分时就不必去重新计算整个树的校验和。
|
||||||
|
|
||||||
|
然后再计算和存储处于顶级的树的校验和,这样如果仓库的任何一部分发生变化,校验和将不可避免地发生变化。
|
||||||
|
|
||||||
|
###### 提交哈希
|
||||||
|
|
||||||
|
一旦创建了树哈希,git 将计算提交哈希,它将包含有关仓库和变更的下列信息:
|
||||||
|
|
||||||
|
* 树哈希的校验和
|
||||||
|
* 变更前树哈希的校验和(父级)
|
||||||
|
* 有关作者的信息(名字、email、创作时间)
|
||||||
|
* 有关提交者的信息(名字、email、提交时间)
|
||||||
|
* 提交信息
|
||||||
|
|
||||||
|
###### 哈希函数
|
||||||
|
|
||||||
|
在写这篇文章时,虽然研究一种更强大的、抗碰撞的算法的工作正在进行,但 git 仍然使用的是 SHA1 哈希机制去计算校验和。注意,git 已经包含了碰撞防范程序,因此认为对 git 成功进行碰撞攻击仍然是不可行的。
|
||||||
|
|
||||||
|
#### 带注释标签和标签签名
|
||||||
|
|
||||||
|
在每个 Git 仓库中,标签允许开发者标记特定的提交。标签可以是 “轻量级的” —— 几乎只是一个特定提交上的指针,或者它们可以是 “带注释的”,它自己将成为 git 树中的项目。一个带注释标签对象包含所有下列的信息:
|
||||||
|
|
||||||
|
* 成为标签的提交的哈希的校验和
|
||||||
|
* 标签名字
|
||||||
|
* 关于打标签的人的信息(名字、email、打标签时间)
|
||||||
|
* 标签信息
|
||||||
|
|
||||||
|
一个 PGP 签名的标签是一个带有将所有这些条目封装进一个 PGP 签名的带注释标签。当开发者签名他们的 git 标签时,他们实际上是向你保证了如下的信息:
|
||||||
|
|
||||||
|
* 他们是谁(以及他们为什么应该被信任)
|
||||||
|
* 他们在签名时的仓库状态是什么样:
|
||||||
|
* 标签包含的提交的哈希
|
||||||
|
* 提交的哈希包含了顶级树的哈希
|
||||||
|
* 顶级树哈希包含了所有文件、内容和子树的哈希
|
||||||
|
* 它也包含有关作者的所有信息
|
||||||
|
* 包含变更发生时的精确时间
|
||||||
|
|
||||||
|
当你克隆一个仓库并验证一个签名的标签时,就是向你以密码方式保证:仓库中的所有内容、包括所有它的历史,与开发者签名时在它的计算机上的仓库完全一致。
|
||||||
|
|
||||||
|
#### 签名的提交
|
||||||
|
|
||||||
|
签名的提交与签名的标签非常类似 —— PGP 签名的是提交对象的内容,而不是标签对象的内容。一个提交签名也给你提供了开发者签名时开发者树上的全部可验证信息。标签签名和提交的 PGP 签名提供了有关仓库和它的完整历史的完全一致的安全保证。
|
||||||
|
|
||||||
|
#### 签名的推送
|
||||||
|
|
||||||
|
为了完整起见,在这里包含了签名的推送这一功能,因为在你使用这个功能之前,需要在接收推送的服务器上先启用它。正如我们在上面所说过的,PGP 签名一个 git 对象就是提供了开发者的 git 树当时的可验证信息,但不提供开发者对那个树意图相关的信息。
|
||||||
|
|
||||||
|
比如,你可以在你自己复刻的 git 仓库的一个实验分支上尝试一个很酷的特性,为了评估它,你提交了你的工作,但是有人在你的代码中发现了一个恶意的 bug。由于你的提交是经过正确签名的,因此有人可能将包含有恶意 bug 的分支推入到 master 分支中,从而在生产系统中引入一个漏洞。由于提交是经过你的密钥正确签名的,所以一切看起来都是合理合法的,而当 bug 被发现时,你的声誉就会因此而受到影响。
|
||||||
|
|
||||||
|
在 `git push` 时,为了验证提交的意图而不仅仅是验证它的内容,添加了要求 PGP 推送签名的功能。
|
||||||
|
|
||||||
|
#### 配置 git 使用你的 PGP 密钥
|
||||||
|
|
||||||
|
如果在你的钥匙环上只有一个密钥,那么你就不需要再做额外的事了,因为它是你的默认密钥。
|
||||||
|
|
||||||
|
然而,如果你有多个密钥,那么你必须要告诉 git 去使用哪一个密钥。(`[fpr]` 是你的密钥的指纹):
|
||||||
|
|
||||||
|
```
|
||||||
|
$ git config --global user.signingKey [fpr]
|
||||||
|
```
|
||||||
|
|
||||||
|
注意:如果你有一个不同的 `gpg2` 命令,那么你应该告诉 git 总是去使用它,而不是传统的版本 1 的 `gpg`:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ git config --global gpg.program gpg2
|
||||||
|
```
|
||||||
|
|
||||||
|
#### 如何使用签名标签
|
||||||
|
|
||||||
|
创建一个签名的标签,只要传递一个简单地 `-s` 开关给 `tag` 命令即可:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ git tag -s [tagname]
|
||||||
|
```
|
||||||
|
|
||||||
|
我们建议始终对 git 标签签名,这样让其它的开发者确信他们使用的 git 仓库没有被恶意地修改过(比如,引入后门):
|
||||||
|
|
||||||
|
##### 如何验证签名的标签
|
||||||
|
|
||||||
|
验证一个签名的标签,只需要简单地使用 `verify-tag` 命令即可:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ git verify-tag [tagname]
|
||||||
|
```
|
||||||
|
|
||||||
|
如果你要验证其他人的 git 标签,那么就需要你导入他的 PGP 公钥。请参考 “可信任的团队沟通” 一文中关于此主题的指导。
|
||||||
|
|
||||||
|
##### 在拉取时验证
|
||||||
|
|
||||||
|
如果你从项目仓库的其它复刻中拉取一个标签,git 将自动验证签名,并在合并操作时显示结果:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ git pull [url] tags/sometag
|
||||||
|
```
|
||||||
|
|
||||||
|
合并信息将包含类似下面的内容:
|
||||||
|
|
||||||
|
```
|
||||||
|
Merge tag 'sometag' of [url]
|
||||||
|
|
||||||
|
[Tag message]
|
||||||
|
|
||||||
|
# gpg: Signature made [...]
|
||||||
|
# gpg: Good signature from [...]
|
||||||
|
```
|
||||||
|
|
||||||
|
#### 配置 git 始终签名带注释标签
|
||||||
|
|
||||||
|
很可能的是,你正在创建一个带注释标签,你应该去签名它。强制 git 始终签名带注释的标签,你可以设置一个全局配置选项:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ git config --global tag.forceSignAnnotated true
|
||||||
|
```
|
||||||
|
|
||||||
|
或者,你始终记得每次都传递一个 `-s` 开关:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ git tag -asm "Tag message" tagname
|
||||||
|
```
|
||||||
|
|
||||||
|
#### 如何使用签名的提交
|
||||||
|
|
||||||
|
创建一个签名的提交很容易,但是将它纳入到你的工作流中却很困难。许多项目使用签名的提交作为一种 “Committed-by:” 的等价行,它记录了代码来源 —— 除了跟踪项目历史外,签名很少有人去验证。在某种意义上,签名的提交用于 “篡改证据”,而不是 git 工作流的 “篡改证明”。
|
||||||
|
|
||||||
|
为创建一个签名的提交,你只需要 `git commit` 命令传递一个 `-S` 标志即可(由于它与另一个标志冲突,所以改为大写的 `-S`):
|
||||||
|
|
||||||
|
```
|
||||||
|
$ git commit -S
|
||||||
|
```
|
||||||
|
|
||||||
|
我们建议始终使用签名提交,并要求项目所有成员都这样做,这样其它人就可以验证它们(下面就讲到如何验证)。
|
||||||
|
|
||||||
|
##### 如何去验证签名的提交
|
||||||
|
|
||||||
|
验证签名的提交需要使用 `verify-commit` 命令:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ git verify-commit [hash]
|
||||||
|
```
|
||||||
|
|
||||||
|
你也可以查看仓库日志,要求所有提交签名是被验证和显示的:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ git log --pretty=short --show-signature
|
||||||
|
```
|
||||||
|
|
||||||
|
##### 在 git 合并时验证提交
|
||||||
|
|
||||||
|
如果项目的所有成员都签名了他们的提交,你可以在合并时强制进行签名检查(然后使用 `-S` 标志对合并操作本身进行签名):
|
||||||
|
|
||||||
|
```
|
||||||
|
$ git merge --verify-signatures -S merged-branch
|
||||||
|
```
|
||||||
|
|
||||||
|
注意,如果有一个提交没有签名或验证失败,将导致合并操作失败。通常情况下,技术是最容易的部分 —— 而人的因素使得项目中很难采用严格的提交验证。
|
||||||
|
|
||||||
|
##### 如果你的项目在补丁管理上采用邮件列表
|
||||||
|
|
||||||
|
如果你的项目在提交和处理补丁时使用一个邮件列表,那么一般很少使用签名提交,因为通过那种方式发送时,签名信息将会丢失。对提交进行签名仍然是非常有用的,这样其他人就能引用你托管在公开 git 树作为参考,但是上游项目接收你的补丁时,仍然不能直接使用 git 去验证它们。
|
||||||
|
|
||||||
|
尽管,你仍然可以签名包含补丁的电子邮件。
|
||||||
|
|
||||||
|
#### 配置 git 始终签名提交
|
||||||
|
|
||||||
|
你可以告诉 git 总是签名提交:
|
||||||
|
|
||||||
|
```
|
||||||
|
git config --global commit.gpgSign true
|
||||||
|
```
|
||||||
|
|
||||||
|
或者你每次都记得给 `git commit` 操作传递一个 `-S` 标志(包括 `—amend`)。
|
||||||
|
|
||||||
|
#### 配置 gpg-agent 选项
|
||||||
|
|
||||||
|
GnuPG agent 是一个守护工具,它能在你使用 gpg 命令时随时自动启动,并运行在后台来缓存私钥的密码。这种方式让你只需要解锁一次密钥就可以重复地使用它(如果你需要在一个自动脚本中签署一组 git 操作,而不想重复输入密钥,这种方式就很方便)。
|
||||||
|
|
||||||
|
为了调整缓存中的密钥过期时间,你应该知道这两个选项:
|
||||||
|
|
||||||
|
* `default-cache-ttl`(秒):如果在 TTL 过期之前再次使用同一个密钥,这个倒计时将重置成另一个倒计时周期。缺省值是 600(10 分钟)。
|
||||||
|
* `max-cache-ttl`(秒):自首次密钥输入以后,不论最近一次使用密钥是什么时间,只要最大值的 TTL 倒计时过期,你将被要求再次输入密码。它的缺省值是 30 分钟。
|
||||||
|
|
||||||
|
如果你认为这些缺省值过短(或过长),你可以编辑 `~/.gnupg/gpg-agent.conf` 文件去设置你自己的值:
|
||||||
|
|
||||||
|
```
|
||||||
|
# set to 30 minutes for regular ttl, and 2 hours for max ttl
|
||||||
|
default-cache-ttl 1800
|
||||||
|
max-cache-ttl 7200
|
||||||
|
```
|
||||||
|
|
||||||
|
##### 补充:与 ssh 一起使用 gpg-agent
|
||||||
|
|
||||||
|
如果你创建了一个 [A](验证)密钥,并将它移到了智能卡,你可以将它用到 ssh 上,为你的 ssh 会话添加一个双因子验证。为了与 agent 沟通你只需要告诉你的环境去使用正确的套接字文件即可。
|
||||||
|
|
||||||
|
首先,添加下列行到你的 `~/.gnupg/gpg-agent.conf` 文件中:
|
||||||
|
|
||||||
|
```
|
||||||
|
enable-ssh-support
|
||||||
|
```
|
||||||
|
|
||||||
|
接着,添加下列行到你的 `.bashrc` 文件中:
|
||||||
|
|
||||||
|
```
|
||||||
|
export SSH_AUTH_SOCK=$(gpgconf --list-dirs agent-ssh-socket)
|
||||||
|
```
|
||||||
|
|
||||||
|
为了让改变生效,你需要杀掉正在运行的 gpg-agent 进程,并重新启动一个新的登入会话:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ killall gpg-agent
|
||||||
|
$ bash
|
||||||
|
$ ssh-add -L
|
||||||
|
```
|
||||||
|
|
||||||
|
最后的命令将列出代表你的 PGP Auth 密钥的 SSH(注释应该会在结束的位置显示: cardno:XXXXXXXX,表示它来自智能卡)。
|
||||||
|
|
||||||
|
为了启用 ssh 的基于密钥的登入,只需要在你要登入的远程系统上添加 `ssh-add -L` 的输出到 `~/.ssh/authorized_keys` 中。祝贺你,这将使你的 SSH 登入凭据更难以窃取。
|
||||||
|
|
||||||
|
此外,你可以从公共密钥服务器上下载其它人的基于 PGP 的 ssh 公钥,这样就可以赋予他登入 ssh 的权利:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ gpg --export-ssh-key [keyid]
|
||||||
|
```
|
||||||
|
|
||||||
|
如果你有让开发人员通过 ssh 来访问 git 仓库的需要,这将让你非常方便。下一篇文章,我们将提供像保护你的密钥那样保护电子邮件帐户的小技巧。
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: https://www.linux.com/blog/learn/pgp/2018/3/protecting-code-integrity-pgp-part-6-using-pgp-git
|
||||||
|
|
||||||
|
作者:[KONSTANTIN RYABITSEV][a]
|
||||||
|
译者:[qhwdw](https://github.com/qhwdw)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]:https://www.linux.com/users/mricon
|
||||||
|
[1]:https://linux.cn/article-9524-1.html
|
||||||
|
[2]:https://linux.cn/article-9529-1.html
|
||||||
|
[3]:https://linux.cn/article-9607-1.html
|
||||||
|
[4]:https://linux.cn/article-10402-1.html
|
||||||
|
[5]:https://linux.cn/article-10415-1.html
|
||||||
|
[6]:https://github.com/jayphelps/git-blame-someone-else
|
@ -0,0 +1,110 @@
|
|||||||
|
用 PGP 保护代码完整性(七):保护在线帐户
|
||||||
|
======
|
||||||
|
|
||||||
|
> 在这个系列的最后一篇当中,我们将为你展示如何用双因子认证保护你的在线账户。
|
||||||
|
|
||||||
|
![](https://www.linux.com/sites/lcom/files/styles/rendered_file/public/online-pgp.jpg?itok=BWc_Bk6q)
|
||||||
|
|
||||||
|
到目前为止,本系列教程已经提供了 PGP 的实用指南,包括基本概念和工具、生成和保护你的密钥的步骤。如果你错过了前面的文章,可以通过下面的链接查看。在本系列的最后一篇文章中,我们将为你保护在线帐户提供一个额外的指南,保护在线帐户是当今非常重要的一件事情。
|
||||||
|
|
||||||
|
- [第一部分:基本概念和工具][1]
|
||||||
|
- [第二部分:生成你的主密钥][2]
|
||||||
|
- [第三部分:生成 PGP 子密钥][3]
|
||||||
|
- [第四部分:将主密钥移到离线存储中][4]
|
||||||
|
- [第五部分:将子密钥移到硬件设备中][5]
|
||||||
|
- [第六部分:在 Git 中使用 PGP][6]
|
||||||
|
|
||||||
|
#### 清单
|
||||||
|
|
||||||
|
* 取得一个具备 U2F 功能的设备(必要)
|
||||||
|
* 为你的在线帐户启用双因子认证(必要)
|
||||||
|
* GitHub/GitLab
|
||||||
|
* Google
|
||||||
|
* 社交媒体
|
||||||
|
* 使用 U2F 作为主验证机制,使用 TOTP 作为备选(必要)
|
||||||
|
|
||||||
|
#### 考虑事项
|
||||||
|
|
||||||
|
你可能注意到,很多在线开发者身份是捆绑了 email 地址的。如果有人能够访问你的邮箱,他们就能够去做一些对你会产生危害的事情,进而会损害你作为自由软件开发者的声誉。应该像保护你的 PGP 密钥那样保护你的 email 地址。
|
||||||
|
|
||||||
|
##### 使用 Fido U2F 的双因子认证
|
||||||
|
|
||||||
|
[双因子认证][7] 是一种提升帐户安全性的机制,它除了要求用户名和密码之外,还要求一个物理令牌。它的目标是即便在有人窃取了你的密码(通过按键记录器、肩窥攻击或其它方式)的情况下,仍然能确保你的帐户安全,他们在没有得到你的一个专用的物理设备(“必备”的那个因子)的情况下,始终不能获取你的帐户。
|
||||||
|
|
||||||
|
广为人知的双因子认证机制有:
|
||||||
|
|
||||||
|
* 基于 SMS 的验证
|
||||||
|
* 借助智能手机应用程序的基于时间的一次性令牌(TOTP),比如 Google Authenticator 或类似解决方案
|
||||||
|
* 支持 Fido U2F 的硬件令牌
|
||||||
|
|
||||||
|
基于 SMS 的验证很容易配置,但是它有如下的缺点:它在没有手机信号的地方无法使用(比如,建筑物的地下室),并且如果攻击者能够阻断或转向 SMS 信息,这种方式可能就会失败,比如通过克隆你的 SIM 卡。
|
||||||
|
|
||||||
|
基于 TOTP 的多因子认证提供了比 SMS 更好的安全保护,但它也有一些重要的缺点(你要在智能手机中添加的那么多令牌中找到正确的那个)。此外,还不能避免一个事实,那就是你的密钥最终还是保存在你的智能手机中 —— 它是一个复杂的、全球连接的设备,它有可能还没有及时从制造商那儿收到安全补丁。
|
||||||
|
|
||||||
|
更重要的是,不论是使用 TOTP 还是 SMS 的方法来保护你免受诱骗攻击 —— 如果诱骗攻击者能够窃取你的帐户密码和双因子令牌,他们就可以在合法的站点上使用它们,访问你的帐户。
|
||||||
|
|
||||||
|
[Fido U2F][8] 是一个按标准开发的专用设备,它能够提供双因子认证机制来对付诱骗攻击。U2F 协议在 USB 令牌中保存每个站点的的唯一密钥,如果你在任何合法站点以外的地方尝试使用它,它将阻止你,以防范偶然让攻击者获得你的密码和一次性令牌。
|
||||||
|
|
||||||
|
Chrome 和 Firefox 都支持 U2F 双因子认证,希望其它浏览器也能够提供对 U2F 的支持。
|
||||||
|
|
||||||
|
#### 获得一个支持 Fido U2F 功能的令牌
|
||||||
|
|
||||||
|
支持 U2F 的硬件令牌的 [可选目标很多][9],但如果你已经订购了一个支持智能卡的物理设备,那么你最好的选择就是 Yubikey 4,它两者都支持。
|
||||||
|
|
||||||
|
#### 启用你的在线帐户的双因子认证
|
||||||
|
|
||||||
|
你要确定想启用双因子认证的在线账户,你的 email 提供商已经使用了(特别是 Google,它对 U2F 的支持非常好)。其它的站点这个功能应该是启用了:
|
||||||
|
|
||||||
|
* GitHub:当你上传你的 PGP 公钥时,你应该要想到,如果其他人能够获得访问你的帐户,他们可以用他们自己的 PGP 公钥替换掉你的 PGP 公钥。如果在 GitHub 上发布代码,你应该使用 U2F 认证来保护你的帐户安全。
|
||||||
|
* GitLab:理由同上
|
||||||
|
* Google:如果你有 google 帐户,你就惊奇地发现,许多帐户都允许以 Google 帐户来代替站点专用的认证来登入它们。
|
||||||
|
* Facebook:理由同上,许多在线站点都提供一个选择让你以 Facebook 的帐户来认证。即便你不使用 Facebook 也应该使用双因子认证来保护你的 Facebook 帐户。
|
||||||
|
* 你认为必要的其它站点。查看 [dongleauth.info][10] 去找找灵感。
|
||||||
|
|
||||||
|
#### 如有可能,配置 TOTP 作为备选
|
||||||
|
|
||||||
|
许多站点都允许你配置多个双因子认证机制,推荐的设置是:
|
||||||
|
|
||||||
|
* U2F 令牌作为主认证机制
|
||||||
|
* TOTP 手机应用作为辅助认证机制
|
||||||
|
|
||||||
|
通过这种方式,即便你丢失了你的 U2F 令牌,你仍然能够重新获取对你的帐户的访问。或者,你可以注册多个 U2F 令牌(即:你可以用一个便宜的令牌仅用它做 U2F,并且将它用作备份)。
|
||||||
|
|
||||||
|
### 延伸阅读
|
||||||
|
|
||||||
|
到目前为止,你已经完成了下列的重要任务:
|
||||||
|
|
||||||
|
1. 创建你的开发者身份并使用 PGP 加密来保护它。
|
||||||
|
2. 通过将你的主密钥移到一个离线存储中并将子密钥移到一个外置硬件设备中的方式来配置你的环境,让窃取你的身份变得极为困难。
|
||||||
|
3. 配置你的 Git 环境去确保任何使用你项目的人都能够验证仓库的完整性和它的整个历史。
|
||||||
|
4. 使用双因子认证强化你的在线帐户。
|
||||||
|
|
||||||
|
在安全保护方面,你已经做的很好了,但是你还应该去阅读以下的主题:
|
||||||
|
|
||||||
|
* 如何去强化你的团队沟通。你的项目开发和治理决策的要求应该和保护提交代码那样去保护,如果不这样做,应该确保你的团队沟通是可信任的,并且所有决策的完整性是可验证的。
|
||||||
|
* 如何去强化你的工作站的安全。你的目标是尽可能减少导致项目代码被污染的危险或你的开发者身份被窃的行为。
|
||||||
|
* 如何写出安全的代码(查看相关编程语言和你项目所使用的库的各种文档)。即便引入它的提交代码上有一个 PGP 签名,糟糕的、不安全的代码仍然是糟糕的、不安全的代码!
|
||||||
|
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: https://www.linux.com/blog/learn/pgp/2018/3/protecting-code-integrity-pgp-part-7-protecting-online-accounts
|
||||||
|
|
||||||
|
作者:[Konstantin Ryabitsev][a]
|
||||||
|
选题:[lujun9972](https://github.com/lujun9972)
|
||||||
|
译者:[qhwdw](https://github.com/qhwdw)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]:https://www.linux.com/users/mricon
|
||||||
|
[1]:https://linux.cn/article-9524-1.html
|
||||||
|
[2]:https://linux.cn/article-9529-1.html
|
||||||
|
[3]:https://linux.cn/article-9607-1.html
|
||||||
|
[4]:https://linux.cn/article-10402-1.html
|
||||||
|
[5]:https://linux.cn/article-10415-1.html
|
||||||
|
[6]:https://linux.cn/article-10421-1.html
|
||||||
|
[7]:https://en.wikipedia.org/wiki/Multi-factor_authentication
|
||||||
|
[8]:https://en.wikipedia.org/wiki/Universal_2nd_Factor
|
||||||
|
[9]:http://www.dongleauth.info/dongles/
|
||||||
|
[10]:http://www.dongleauth.info/
|
@ -0,0 +1,86 @@
|
|||||||
|
使用 Xfce Linux 桌面环境的 8 个理由
|
||||||
|
============================
|
||||||
|
> 整体上很优雅的 Xfce 桌面所具备的足够轻巧和快速的特性能够让它很容易地知道如何做好一件事。
|
||||||
|
|
||||||
|
![](https://opensource.com/sites/default/files/styles/image-full-size/public/lead-images/linux_penguin_green.png?itok=ENdVzW22)
|
||||||
|
|
||||||
|
由于某些原因(也包括好奇),几周前我开始使用 [Xfce][1] 作为我的 Linux 桌面。促使我更换 Linux 桌面环境的原因之一是桌面相关的守护进程占据了我的性能非常强大的主工作站的绝大部分 CPU 资源和 I/O 带宽。当然,有些不稳定性可能是因为我删除了提供这些守护进程的 RPM 包。然而,事实是在我删除这些 RPM 包之前,KDE 就已经很不稳定了而且还导致了一系列其他方面的性能和稳定性问题。所以我需要换一个桌面来避免这些问题。
|
||||||
|
|
||||||
|
在回顾了我为 Linux 桌面所写的一系列文章后我才意识到我忽略了 Xfce。这篇文章也是力图能够纠正弥补这个疏忽。我非常喜欢 Xfce 也很享受它所带给我超乎预期的快速、轻量的体验。
|
||||||
|
|
||||||
|
作为研究的一部分,我有尝试过在 Google 上查询 Xfce 对应什么意思。有个历史参考是它对应着 “XForms Common Environment”,但 Xfce 早已不在使用 XForms 工具。几年前,我找到另一个参考是 “Xtra fine computing environment” 而且我也很喜欢这个解释。我将会用它作为 Xfce 的全称(尽管再也找不到这个参考页面)。
|
||||||
|
|
||||||
|
### 推荐 Xfce 的 8 个理由
|
||||||
|
|
||||||
|
#### 1、轻量级架构
|
||||||
|
|
||||||
|
Xfce 相对于其他的桌面如 KDE 和 GNOME,不管是内存还是 CPU 的占用率都非常小。在我的系统中,组成 Xfce 桌面的程序仅占用了少量内存就构成一个如此强大的桌面。超低的 CPU 占用率也是 Xfce 桌面的一个特点。了解到 Xfce 内存占用特别低后,我对它的 CPU 占用率也非常低这个特性自然而言也就不感到奇怪了。
|
||||||
|
|
||||||
|
#### 2、简洁
|
||||||
|
|
||||||
|
Xfce 桌面很简单,就像绒毛整洁的动物让人一目了然、赏心悦目。基础的桌面有两个面板和一条在左边垂直的图标行。面板 0 在底部,并由一些基础的应用启动程序和能访问到系统里对应程序的图标组成。面板 1 在顶部,由一个应用程序启动器和一个能够允许用户在多个工作区之间来回切换的工作区切换器组成。面板可以通过补充项自定义修改,比如增加个新的应用启动器或者更改它们的宽高。
|
||||||
|
|
||||||
|
桌面左侧的图标对应是家目录和垃圾桶。它也可以显示其他的图标,如完整的文件系统目录树和已连接上系统的可插拔的任意 USB 存储设备。这些图标可以用来挂载和卸载设备,也可以用来打开默认的文件管理器。如果你愿意,它们都可以被隐藏,同时文件系统、垃圾箱、家目录对应的图标都可以逐个控制管理。所有的可移动设备也可以被隐藏或作为一个组显示。
|
||||||
|
|
||||||
|
#### 3、文件管理
|
||||||
|
|
||||||
|
作为 Xfce 的默认文件管理器 Thunar,它很简单,既易于使用和配置也非常容易学习。尽管它并不像其他的文件管理器比如 Konqueror 或者 Dolphin 那样效果华丽,但它很强大也很快。Thunar 并不能在一个窗口里面打开多个面板,但它提供了选项卡来支持多个目录的同时打开。Thunar 也有一个非常漂亮的侧边栏,其上同样的图标就像桌面那样能够显示完整的文件系统目录树和所有已连接的 USB 存储设备。设备能够被挂载和卸载,可移动媒介如 CD 也能够被弹出。Thunar 也可以使用类似 Ark 这种帮助软件来在你点击归档文件的时候打开它们。比如 ZIP、TAR、RPM 这种归档文件都可以被浏览也可以从中复制单个文件。
|
||||||
|
|
||||||
|
![Xfce desktop][3]
|
||||||
|
|
||||||
|
*Xfce 桌面及 Thunar 和 Xfce 下的终端模拟器。*
|
||||||
|
|
||||||
|
在我的[文件管理器系列][4]文章中,我已经使用体验过很多不同的文件管理器软件,我不得不说 Thunar 的简单易用让你无法不喜欢上它。它可以让你通过使用侧边栏来很容易地浏览文件系统。
|
||||||
|
|
||||||
|
#### 4、稳定
|
||||||
|
|
||||||
|
Xfce 桌面非常稳定。新版本的发布周期似乎是三年,但也会根据需要发布相关更新。最新的版本是于 2015 年 2 月发布的 4.12。在使用 KDE 遇到一系列问题后,稳如磐石的 Xfce 桌面环境显得让人格外放心。在我使用 Xfce 的过程中,它从来没有崩溃过,也不会产生额外的守护进程占据过多的系统资源。这正是我想要的:它安安静静地工作,不会给你带来额外的困扰。
|
||||||
|
|
||||||
|
#### 5、优雅
|
||||||
|
|
||||||
|
Xfce 简单优雅。在我的今年秋天将面世的新书《系统管理员的 Linux 哲学》中我谈到了关于简单的一系列好处,包括简单事实上也是优雅的诸多标志之一。很明确能够确定的就是 Xfce 及相关组件程序的开发者和维护者也是极力推崇简单至上。这种简单特性很可能也是 Xfce 如此稳定的主要原因,但它也给用户带来了一个整洁的桌面外观,一个反应灵敏的操作界面,一个会让人感觉很自然也很易用的导航结构,而且 Xfce 整体上的优雅特性也会让用户的使用过程中充满愉悦感。
|
||||||
|
|
||||||
|
#### 6、终端仿真程序
|
||||||
|
|
||||||
|
Xfce4 的终端仿真程序非常强大,而且和其他很多终端仿真程序一样可以允许你使用多个选项卡来让多个终端在一个单独窗口里共存。尽管它与 Tilix、Terminator、Konsole 这种终端仿真程序比起来相对简陋,但它也能很好的完成工作。选项卡的名字可以更改,而且选项卡也可以通过拖放或者工具栏的箭头图标或者菜单栏的选项重新排列。我特别喜欢 Xfce 的终端仿真程序的一点就是不管你连接了多少主机,相对应的选项卡都会显示对应的主机名,比如,从 host1=>host2=>host3=>host4,会准确地在选项卡显示了 “host4”。但其他的终端仿真程序最多也就显示 “host2”。
|
||||||
|
|
||||||
|
至于这个终端仿真程序功能和外观的其他方面都可以根据你的需要很容易配置成你想要的。当然同 Xfce 的其他组件一样,这款终端仿真程序占用了系统资源的很少一部分。
|
||||||
|
|
||||||
|
#### 7、可配置性
|
||||||
|
|
||||||
|
Xfce 能够配置的范围极大。虽然 Xfce 桌面的可配置性比不上 KDE,但依旧远超 GNOME,而且比它更容易配置。比如,我发现设置管理器是 Xfce 配置一切的入口。虽然每个配置程序都可以单独使用,但是设置管理器把他们都放在一个窗口里以便快速访问。关于 Xfce 桌面很多重要的部分都可以通过配置来满足我的需求。
|
||||||
|
|
||||||
|
#### 8、模块化
|
||||||
|
|
||||||
|
Xfce 是由一系列单个的项目组成的整体,而且在你的 Linux 桌面发行版中也未必安装了 Xfce 的所有组件。[Xfce 项目][5] 的主页列出了主要的项目,所以你可以根据需要安装你想安装的附加组件。比如在我的 Fedora 28 workstation 版本上我安装的 Xfce 组时就没有 [Xfce 项目][5] 主页最下面的说明的一些程序。
|
||||||
|
|
||||||
|
这里还有个关于 Xfce 的 [文档页面][6] 和 一个被称为 [Xfce 超值项目][7] 的 wiki 列举了其他的 Xfce 相关的项目,它们为 Xfce 的面板及 Thunar 提供了很多不错的应用程序、精美的插图、好用的插件。
|
||||||
|
|
||||||
|
### 总结
|
||||||
|
|
||||||
|
整体上很优雅的 Xfce 桌面所具备的足够轻巧和快速的特性能够让它很容易地知道如何做好一件事。它的轻量级的结构也节省了大量的 CPU 和内存资源。这也使得 Xfce 非常适合那种由于硬件有限而无法分配给桌面太多资源的旧主机。然而,Xfce 又是足够的灵活和强大的,能够满足高级用户的需要。
|
||||||
|
|
||||||
|
我知道,更换到一个新的 Linux 桌面环境需要你自己按照你想要的做些对应的自定义设置:比如面板上显示你最爱用的程序对应的启动器,设置下你最喜欢的桌面背景壁纸等一系列工作。这些年来我已经在切换到新桌面环境或更新旧桌面环境折腾很多次了。这需要时间也需要耐心。
|
||||||
|
|
||||||
|
我觉得切换 Linux 的桌面环境就像我在工作中换个办公工位或者办公室一样。别人把我的东西装箱从旧办公室搬到新办公室,然后我在我的新办公室里组装连接好我的电脑,打开箱子再把里面的东西放在合适的位置。而切换到 Xfce 桌面大概就是我做过的最简单省事容易的桌面环境更换了。
|
||||||
|
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: https://opensource.com/article/18/6/xfce-desktop
|
||||||
|
|
||||||
|
作者:[David Both][a]
|
||||||
|
选题:[lujun9972](https://github.com/lujun9972)
|
||||||
|
译者:[WangYueScream](https://github.com/WangYueScream)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]:https://opensource.com/users/dboth
|
||||||
|
[1]:https://xfce.org/
|
||||||
|
[2]:/file/401856
|
||||||
|
[3]:https://opensource.com/sites/default/files/uploads/xfce-desktop-01.png (Xfce desktop)
|
||||||
|
[4]:https://opensource.com/sitewide-search?search_api_views_fulltext=David%20Both%20File%20managers
|
||||||
|
[5]:https://xfce.org/projects
|
||||||
|
[6]:https://docs.xfce.org/
|
||||||
|
[7]:https://goodies.xfce.org/
|
@ -0,0 +1,99 @@
|
|||||||
|
10 个供管理员救急的杀手级工具
|
||||||
|
======
|
||||||
|
|
||||||
|
> 可以让你赶快离开办公室的网络管理技巧和工具。
|
||||||
|
|
||||||
|
![](https://opensource.com/sites/default/files/styles/image-full-size/public/lead-images/cloud_tools_hardware.png?itok=PGjJenqT)
|
||||||
|
|
||||||
|
当工作任务堆积成山时,管理网络和系统就变得十分有压力了。没有人能真正意识到需要花费多长时间,每个人都希望在昨天就完成他们的工作。
|
||||||
|
|
||||||
|
所以难怪我们这么多人都被致力于找出有效的方法并与大家分享的开源精神所吸引。因为,当截止日期来临,并且当天没有足够多的时间时,如果你可以找到立刻施行的免费答案,那会非常有帮助。
|
||||||
|
|
||||||
|
因此,闲话少叙,下述是我的瑞士军刀,可以保证你在晚饭前离开办公室。
|
||||||
|
|
||||||
|
### 服务器配置和脚本
|
||||||
|
|
||||||
|
让我们看一看!
|
||||||
|
|
||||||
|
- [NixCraft][1]
|
||||||
|
|
||||||
|
使用该网站的搜索功能。经过十多年的定期更新,这里遍地是黄金!有用的脚本和方便的技巧可以立刻解决你的问题。这是我一般使用 Google 后的第二个选项。
|
||||||
|
|
||||||
|
- [Webmin][2]
|
||||||
|
|
||||||
|
它提供给你了一个很好的 Web 界面来帮助你远程编辑配置文件。它减少了在处理目录路径和 `sudo nano` 上花费的大量时间,在你处理多个客户时,非常方便。
|
||||||
|
|
||||||
|
- [Windows 下的 Linux 子系统][3]
|
||||||
|
|
||||||
|
现代工作场所的现实是大多数员工都运行着 Windows,而服务器机房中不断增长的设备则运行着 Linux 。因此,有些时候你会发现尝试在 Windows 桌面上执行管理任务。
|
||||||
|
|
||||||
|
你怎么做?装一个虚拟机?如果安装目前 Windows 10 中免费提供的 Linux 子系统的兼容层,实际上要快得多,配置要少的多。
|
||||||
|
|
||||||
|
这为你提供了一个 Bash 终端窗口,你可以在这个窗口中执行本地计算机上的 Bash 脚本和 Linux 二进制文件,可以完全访问 Windows 和 Linux 文件系统,以及安装网络驱动器。它包含 Ubuntu 、OpenSUSE、SLES、Debian 和 Kali 发行版。
|
||||||
|
|
||||||
|
- [mRemoteNG][4]
|
||||||
|
|
||||||
|
当你有 100 多个服务器需要去管理时,这会是一个出色的 SSH 和远程桌面客户端。
|
||||||
|
|
||||||
|
### 设置网络,这样你就无需再这样做了。
|
||||||
|
|
||||||
|
一个设计不周的网络是厌恶加班的管理员的死敌。
|
||||||
|
|
||||||
|
- [可拓展的 IP 寻址方案][5]
|
||||||
|
|
||||||
|
IP 地址耗尽的可怕之处在于,当 IP 地址耗尽时,网络已经变的足够大,而新的寻址方案是众所周知的昂贵、令人痛苦的耗时。
|
||||||
|
|
||||||
|
没有人有时间做这件事!
|
||||||
|
|
||||||
|
到了某个时候,IPv6 终将到来,来拯救这世界。但在那之前,无论世界向我们扔了多少可穿戴设备、平板电脑、智能锁、灯、安全摄像头、VoIP 耳机和浓缩咖啡机,这些以不变应万变的 IP 寻址方案都应该让我们继续前行。
|
||||||
|
|
||||||
|
- [Linux Chmod 权限备忘录][6]
|
||||||
|
|
||||||
|
一个简短但是有用的 Bash 命令备忘录可以帮助你通过网络设置权限。所以,客户服务部的账单落入到勒索软件骗局时,你可以只恢复他们的文件,而不是整个公司的文件。
|
||||||
|
|
||||||
|
- [VLSM 子网计算器][7]
|
||||||
|
|
||||||
|
只需要输入你想要从地址空间中创建的网络的数量,以及每个网络所需要的主机数量,它就可以计算出所有的子网掩码应该是什么。
|
||||||
|
|
||||||
|
### 单一用途的 Linux 发行版
|
||||||
|
|
||||||
|
需要一个只做一件事的 Linux 容器?如果其他人已经在一个操作系统上搞好了一个小东西,你就可以快速安装它并马上投入使用。
|
||||||
|
|
||||||
|
下面这些每一个都使得我的工作变得轻松了许多。
|
||||||
|
|
||||||
|
- [Porteus Kiosk][8]
|
||||||
|
|
||||||
|
这个工具用来帮你把一台电脑上锁定到一个浏览器上。通过稍稍一些调整,你甚至可以把浏览器锁定在一个特定的网站上。它对于公共访问机器来说非常方便。它可以与触摸屏或键盘鼠标配合使用。
|
||||||
|
|
||||||
|
- [Parted Magic][9]
|
||||||
|
|
||||||
|
这是一个你可以从 USB 驱动器启动的,可以用来划分磁盘驱动器、恢复数据并运行基准测试工具的操作系统。
|
||||||
|
|
||||||
|
- [IPFire][10]
|
||||||
|
|
||||||
|
啊哈~我还是不敢相信有人把路由器/防火墙/代理组合成为“我尿火”(LCTT 译注:IPFire 和 “I pee Fire“ 同音)。这是我在这个 Linux 发行版中第二喜欢的东西。我最喜欢的是它是一个非常可靠的软件套件,设置和配置十分容易,而且有一系列的插件可以拓展它。
|
||||||
|
|
||||||
|
那么,你呢?你发现了哪些工具、资源和备忘录可以让我们的工作日更加的轻松?我很高兴知道,请在评论中分享您的工具。
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: https://opensource.com/article/18/7/tools-admin
|
||||||
|
|
||||||
|
作者:[Grant Hamono][a]
|
||||||
|
选题:[lujun9972](https://github.com/lujun9972)
|
||||||
|
译者:[bestony](https://github.com/bestony)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]:https://opensource.com/users/grantdxm
|
||||||
|
[1]:https://www.cyberciti.biz/
|
||||||
|
[2]:http://www.webmin.com/
|
||||||
|
[3]:http://wsl-guide.org/en/latest/
|
||||||
|
[4]:https://mremoteng.org/
|
||||||
|
[5]:https://blog.dxmtechsupport.com.au/ip-addressing-for-a-small-business-that-might-grow/
|
||||||
|
[6]:https://isabelcastillo.com/linux-chmod-permissions-cheat-sheet
|
||||||
|
[7]:http://www.vlsm-calc.net/
|
||||||
|
[8]:http://porteus-kiosk.org/
|
||||||
|
[9]:https://partedmagic.com/
|
||||||
|
[10]:https://www.ipfire.org/
|
335
published/20180725 Build an interactive CLI with Node.js.md
Normal file
335
published/20180725 Build an interactive CLI with Node.js.md
Normal file
@ -0,0 +1,335 @@
|
|||||||
|
使用 Node.js 构建交互式命令行工具
|
||||||
|
======
|
||||||
|
|
||||||
|
> 使用 Node.js 构建一个根据询问创建文件的命令行工具。
|
||||||
|
|
||||||
|
![](https://opensource.com/sites/default/files/styles/image-full-size/public/lead-images/programming_keyboard_coding.png?itok=E0Vvam7A)
|
||||||
|
|
||||||
|
当用于构建命令行界面(CLI)时,Node.js 十分有用。在这篇文章中,我将会教你如何使用 [Node.js][1] 来构建一个问一些问题并基于回答创建一个文件的命令行工具。
|
||||||
|
|
||||||
|
### 开始
|
||||||
|
|
||||||
|
首先,创建一个新的 [npm][2] 包(NPM 是 JavaScript 包管理器)。
|
||||||
|
|
||||||
|
```
|
||||||
|
mkdir my-script
|
||||||
|
cd my-script
|
||||||
|
npm init
|
||||||
|
```
|
||||||
|
|
||||||
|
NPM 将会问一些问题。随后,我们需要安装一些包。
|
||||||
|
|
||||||
|
```
|
||||||
|
npm install --save chalk figlet inquirer shelljs
|
||||||
|
```
|
||||||
|
|
||||||
|
这是我们需要的包:
|
||||||
|
|
||||||
|
* Chalk:正确设定终端的字符样式
|
||||||
|
* Figlet:使用普通字符制作大字母的程序(LCTT 译注:使用标准字符,拼凑出图片)
|
||||||
|
* Inquirer:通用交互式命令行用户界面的集合
|
||||||
|
* ShellJS:Node.js 版本的可移植 Unix Shell 命令行工具
|
||||||
|
|
||||||
|
### 创建一个 index.js 文件
|
||||||
|
|
||||||
|
现在我们要使用下述内容创建一个 `index.js` 文件。
|
||||||
|
|
||||||
|
```
|
||||||
|
#!/usr/bin/env node
|
||||||
|
|
||||||
|
const inquirer = require("inquirer");
|
||||||
|
const chalk = require("chalk");
|
||||||
|
const figlet = require("figlet");
|
||||||
|
const shell = require("shelljs");
|
||||||
|
```
|
||||||
|
|
||||||
|
### 规划命令行工具
|
||||||
|
|
||||||
|
在我们写命令行工具所需的任何代码之前,做计划总是很棒的。这个命令行工具只做一件事:**创建一个文件**。
|
||||||
|
|
||||||
|
它将会问两个问题:文件名是什么以及文件后缀名是什么?然后创建文件,并展示一个包含了所创建文件路径的成功信息。
|
||||||
|
|
||||||
|
```
|
||||||
|
// index.js
|
||||||
|
|
||||||
|
const run = async () => {
|
||||||
|
// show script introduction
|
||||||
|
// ask questions
|
||||||
|
// create the file
|
||||||
|
// show success message
|
||||||
|
};
|
||||||
|
|
||||||
|
run();
|
||||||
|
```
|
||||||
|
|
||||||
|
第一个函数只是该脚本的介绍。让我们使用 `chalk` 和 `figlet` 来把它完成。
|
||||||
|
|
||||||
|
```
|
||||||
|
const init = () => {
|
||||||
|
console.log(
|
||||||
|
chalk.green(
|
||||||
|
figlet.textSync("Node JS CLI", {
|
||||||
|
font: "Ghost",
|
||||||
|
horizontalLayout: "default",
|
||||||
|
verticalLayout: "default"
|
||||||
|
})
|
||||||
|
)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
const run = async () => {
|
||||||
|
// show script introduction
|
||||||
|
init();
|
||||||
|
|
||||||
|
// ask questions
|
||||||
|
// create the file
|
||||||
|
// show success message
|
||||||
|
};
|
||||||
|
|
||||||
|
run();
|
||||||
|
```
|
||||||
|
|
||||||
|
然后,我们来写一个函数来问问题。
|
||||||
|
|
||||||
|
```
|
||||||
|
const askQuestions = () => {
|
||||||
|
const questions = [
|
||||||
|
{
|
||||||
|
name: "FILENAME",
|
||||||
|
type: "input",
|
||||||
|
message: "What is the name of the file without extension?"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: "list",
|
||||||
|
name: "EXTENSION",
|
||||||
|
message: "What is the file extension?",
|
||||||
|
choices: [".rb", ".js", ".php", ".css"],
|
||||||
|
filter: function(val) {
|
||||||
|
return val.split(".")[1];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
];
|
||||||
|
return inquirer.prompt(questions);
|
||||||
|
};
|
||||||
|
|
||||||
|
// ...
|
||||||
|
|
||||||
|
const run = async () => {
|
||||||
|
// show script introduction
|
||||||
|
init();
|
||||||
|
|
||||||
|
// ask questions
|
||||||
|
const answers = await askQuestions();
|
||||||
|
const { FILENAME, EXTENSION } = answers;
|
||||||
|
|
||||||
|
// create the file
|
||||||
|
// show success message
|
||||||
|
};
|
||||||
|
```
|
||||||
|
|
||||||
|
注意,常量 `FILENAME` 和 `EXTENSIONS` 来自 `inquirer` 包。
|
||||||
|
|
||||||
|
下一步将会创建文件。
|
||||||
|
|
||||||
|
```
|
||||||
|
const createFile = (filename, extension) => {
|
||||||
|
const filePath = `${process.cwd()}/${filename}.${extension}`
|
||||||
|
shell.touch(filePath);
|
||||||
|
return filePath;
|
||||||
|
};
|
||||||
|
|
||||||
|
// ...
|
||||||
|
|
||||||
|
const run = async () => {
|
||||||
|
// show script introduction
|
||||||
|
init();
|
||||||
|
|
||||||
|
// ask questions
|
||||||
|
const answers = await askQuestions();
|
||||||
|
const { FILENAME, EXTENSION } = answers;
|
||||||
|
|
||||||
|
// create the file
|
||||||
|
const filePath = createFile(FILENAME, EXTENSION);
|
||||||
|
|
||||||
|
// show success message
|
||||||
|
};
|
||||||
|
```
|
||||||
|
|
||||||
|
最后,重要的是,我们将展示成功信息以及文件路径。
|
||||||
|
|
||||||
|
```
|
||||||
|
const success = (filepath) => {
|
||||||
|
console.log(
|
||||||
|
chalk.white.bgGreen.bold(`Done! File created at ${filepath}`)
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
// ...
|
||||||
|
|
||||||
|
const run = async () => {
|
||||||
|
// show script introduction
|
||||||
|
init();
|
||||||
|
|
||||||
|
// ask questions
|
||||||
|
const answers = await askQuestions();
|
||||||
|
const { FILENAME, EXTENSION } = answers;
|
||||||
|
|
||||||
|
// create the file
|
||||||
|
const filePath = createFile(FILENAME, EXTENSION);
|
||||||
|
|
||||||
|
// show success message
|
||||||
|
success(filePath);
|
||||||
|
};
|
||||||
|
```
|
||||||
|
|
||||||
|
|
||||||
|
来让我们通过运行 `node index.js` 来测试这个脚本,这是我们得到的:
|
||||||
|
|
||||||
|
![](https://opensource.com/sites/default/files/uploads/nodejs_output.png)
|
||||||
|
|
||||||
|
### 完整代码
|
||||||
|
|
||||||
|
下述代码为完整代码:
|
||||||
|
|
||||||
|
```
|
||||||
|
#!/usr/bin/env node
|
||||||
|
|
||||||
|
const inquirer = require("inquirer");
|
||||||
|
const chalk = require("chalk");
|
||||||
|
const figlet = require("figlet");
|
||||||
|
const shell = require("shelljs");
|
||||||
|
|
||||||
|
const init = () => {
|
||||||
|
console.log(
|
||||||
|
chalk.green(
|
||||||
|
figlet.textSync("Node JS CLI", {
|
||||||
|
font: "Ghost",
|
||||||
|
horizontalLayout: "default",
|
||||||
|
verticalLayout: "default"
|
||||||
|
})
|
||||||
|
)
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
const askQuestions = () => {
|
||||||
|
const questions = [
|
||||||
|
{
|
||||||
|
name: "FILENAME",
|
||||||
|
type: "input",
|
||||||
|
message: "What is the name of the file without extension?"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: "list",
|
||||||
|
name: "EXTENSION",
|
||||||
|
message: "What is the file extension?",
|
||||||
|
choices: [".rb", ".js", ".php", ".css"],
|
||||||
|
filter: function(val) {
|
||||||
|
return val.split(".")[1];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
];
|
||||||
|
return inquirer.prompt(questions);
|
||||||
|
};
|
||||||
|
|
||||||
|
const createFile = (filename, extension) => {
|
||||||
|
const filePath = `${process.cwd()}/${filename}.${extension}`
|
||||||
|
shell.touch(filePath);
|
||||||
|
return filePath;
|
||||||
|
};
|
||||||
|
|
||||||
|
const success = filepath => {
|
||||||
|
console.log(
|
||||||
|
chalk.white.bgGreen.bold(`Done! File created at ${filepath}`)
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
const run = async () => {
|
||||||
|
// show script introduction
|
||||||
|
init();
|
||||||
|
|
||||||
|
// ask questions
|
||||||
|
const answers = await askQuestions();
|
||||||
|
const { FILENAME, EXTENSION } = answers;
|
||||||
|
|
||||||
|
// create the file
|
||||||
|
const filePath = createFile(FILENAME, EXTENSION);
|
||||||
|
|
||||||
|
// show success message
|
||||||
|
success(filePath);
|
||||||
|
};
|
||||||
|
|
||||||
|
run();
|
||||||
|
```
|
||||||
|
|
||||||
|
### 使用这个脚本
|
||||||
|
|
||||||
|
想要在其它地方执行这个脚本,在你的 `package.json` 文件中添加一个 `bin` 部分,并执行 `npm link`:
|
||||||
|
|
||||||
|
```
|
||||||
|
{
|
||||||
|
"name": "creator",
|
||||||
|
"version": "1.0.0",
|
||||||
|
"description": "",
|
||||||
|
"main": "index.js",
|
||||||
|
"scripts": {
|
||||||
|
"test": "echo \"Error: no test specified\" && exit 1",
|
||||||
|
"start": "node index.js"
|
||||||
|
},
|
||||||
|
"author": "",
|
||||||
|
"license": "ISC",
|
||||||
|
"dependencies": {
|
||||||
|
"chalk": "^2.4.1",
|
||||||
|
"figlet": "^1.2.0",
|
||||||
|
"inquirer": "^6.0.0",
|
||||||
|
"shelljs": "^0.8.2"
|
||||||
|
},
|
||||||
|
"bin": {
|
||||||
|
"creator": "./index.js"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
执行 `npm link` 使得这个脚本可以在任何地方调用。
|
||||||
|
|
||||||
|
这就是是当你运行这个命令时的结果。
|
||||||
|
|
||||||
|
```
|
||||||
|
/usr/bin/creator -> /usr/lib/node_modules/creator/index.js
|
||||||
|
/usr/lib/node_modules/creator -> /home/hugo/code/creator
|
||||||
|
```
|
||||||
|
|
||||||
|
这会连接 `index.js` 作为一个可执行文件。这是完全可能的,因为这个 CLI 脚本的第一行是 `#!/usr/bin/env node`。
|
||||||
|
|
||||||
|
现在我们可以通过执行如下命令来调用。
|
||||||
|
|
||||||
|
```
|
||||||
|
$ creator
|
||||||
|
```
|
||||||
|
|
||||||
|
### 总结
|
||||||
|
|
||||||
|
正如你所看到的,Node.js 使得构建一个好的命令行工具变得非常简单。如果你希望了解更多内容,查看下列包。
|
||||||
|
|
||||||
|
* [meow][3]:一个简单的命令行助手工具
|
||||||
|
* [yargs][4]:一个命令行参数解析工具
|
||||||
|
* [pkg][5]:将你的 Node.js 程序包装在一个可执行文件中。
|
||||||
|
|
||||||
|
在评论中留下你关于构建命令行工具的经验吧!
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: https://opensource.com/article/18/7/node-js-interactive-cli
|
||||||
|
|
||||||
|
作者:[Hugo Dias][a]
|
||||||
|
选题:[lujun9972](https://github.com/lujun9972)
|
||||||
|
译者:[bestony](https://github.com/bestony)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]:https://opensource.com/users/hugodias
|
||||||
|
[1]:https://nodejs.org/en/
|
||||||
|
[2]:https://www.npmjs.com/
|
||||||
|
[3]:https://github.com/sindresorhus/meow
|
||||||
|
[4]:https://github.com/yargs/yargs
|
||||||
|
[5]:https://github.com/zeit/pkg
|
118
published/20180731 How to be the lazy sysadmin.md
Normal file
118
published/20180731 How to be the lazy sysadmin.md
Normal file
@ -0,0 +1,118 @@
|
|||||||
|
如何成为一名懒惰的系统管理员
|
||||||
|
======
|
||||||
|
> 我们是聪明地工作,而不是刻苦工作,但仍能把工作做好。
|
||||||
|
|
||||||
|
![](https://opensource.com/sites/default/files/styles/image-full-size/public/lead-images/cat-yawn-vm.png?itok=0c_zy6aQ)
|
||||||
|
|
||||||
|
Linux 的系统管理员的工作总是复杂的,并且总是伴随着各种陷阱和障碍。做每件事都没有足够时间,当你想完成那个半秃头老板(PHB)给的任务时,他(只会)不停在你的后面盯着,而最核心的服务器总是在最不合时宜的时间点崩溃,问题和挑战比比皆是。而我发现,成为一名<ruby>懒惰的系统管理员<rt>Lazy SysAdmin</rt></ruby>可以解决这一困境。
|
||||||
|
|
||||||
|
(LCTT 译注:<ruby>[半秃头老板](https://en.wikipedia.org/wiki/Pointy-haired_Boss)<rt>Pointy-Haired Boss</rt></ruby>(PHB),那是[呆伯特漫画](https://zh.wikipedia.org/wiki/%E5%91%86%E4%BC%AF%E7%89%B9)中的角色,缺乏一般知识常识及其职位所应具有的管理能力,爱说大话且富有向物理显示挑战的精神,大概长成下图这样。)
|
||||||
|
|
||||||
|
![](https://cdn-images-1.medium.com/max/1600/1*qu5upg6tgVSXx4KqL-4gZw.jpeg)
|
||||||
|
|
||||||
|
我在即将在 Apress 出版的新书 《The Linux Philosophy for SysAdmins》(LCTT 译注:暂译《系统管理员的 Linux 哲学》)中更详细地讨论如何成为一个懒惰的系统管理员,那书预计会在 9 月出版(LCTT 译注:已于 2018 年 8 月出版)。这本的部分内容摘录自该书,特别是第九章,“成为一名懒惰的系统管理员”。在我们讨论如何做到这点前,让我们简单了解一下成为一个名懒惰的系统管理员意味着什么。
|
||||||
|
|
||||||
|
![](https://images-na.ssl-images-amazon.com/images/I/41YOHTGClbL._SX348_BO1,204,203,200_.jpg)
|
||||||
|
|
||||||
|
### 真实生产力 vs. 虚假生产力
|
||||||
|
|
||||||
|
#### 虚假生产力
|
||||||
|
|
||||||
|
在我工作的地方,半秃头老板相信的管理风格叫“<ruby>走动式管理<rt>management by walking around</rt></ruby>”。通过判断某人在不在他的键盘上输入东西,或者至少要看看他们显示器上显示的东西,来判断他们的工作是否有效率。这是一个糟糕的工作场所。各部门间有道很高的行政墙,这会造就了许多的、微小的沟通壁垒,大量无用的文书工作,以及获得任何事情的许可都要等待漫长的时间。因为这样、那样的原因,不可能高效地做任何事情,如果真的是这样,那是非常低效。为了看起来很忙,我们都有自己的一套<ruby>“看起来很忙”的工具包<rt>Look Busy Kits</rt></ruby>(LBK),可能是一些短小的、用来显示一些行为活动的 Bash 脚本,或者是 `top`、`htop`、`iotop` 之类的程序,或者是一些持续显示某些行为活动的监控工具。这种工作场所的风气让人不可能真正高效,我讨厌这种地方,也讨厌那个几乎不可能完成任何有价值的事情的事实。
|
||||||
|
|
||||||
|
这种糟糕场所对真实的系统管理员来讲是场噩梦。没有人会感到快乐。在那里花费四五个月才能完成的事,在其他地方只需的一个早晨。我们没有什么实际工作要做,却要花大量的时间干活来让自己看起来很忙。我们在默默地竞赛,看谁能创造最好的“看起来很忙”的工具包,这就是我们花费最多时间的地方了。那份工作我只做了几个月,但好像已经耗费了一生。如果你看到的这个监狱的表面,你可能会说我们是很懒,因为我们只完成了几乎为 0 的实际工作。
|
||||||
|
|
||||||
|
这是个极端的例子,它完全与我所说的“我是一个懒惰的系统管理员”的意思相反,而做一个懒惰的系统管理是件好事。
|
||||||
|
|
||||||
|
#### 真实生产力
|
||||||
|
|
||||||
|
我很幸运,曾为一些真正的管理者工作过 —— 他们明白,系统管理员的生产力并不是以每天花多少小时敲键盘来衡量。毕竟,即使一只猴子能敲击他们的键盘,但也不能说明结果的价值。
|
||||||
|
|
||||||
|
正如我书中所言:
|
||||||
|
|
||||||
|
> “我是一个懒惰的系统管理员,同时我也是一个高效的系统管理员。这两者看似矛盾的说法不是相互排斥的,而是会以一种非常积极的方式相辅相成……
|
||||||
|
>
|
||||||
|
> “系统管理员在思考的时候是最高效的 —— 思考关于如何解决现有问题和避免未来的问题;思考怎样监控 Linux 计算机,以便找到预测和预示这些未来的问题的线索;思考如何让他们的工作更有效率;思考如何自动化所有这些要执行的任务,无论是每天还是每年一次的任务。
|
||||||
|
>
|
||||||
|
> “系统管理员冥思苦想的那一面是不会被非系统管理员所熟知的,那些人包括很多管理着系统管理员的人,比如那个半秃头老板。系统管理员都会以不同的方式解决他们工作中苦思的部分。一些我认识的系统管理员会在沙滩、骑自行车、参加马拉松或者攀岩时找到最好的想法。另一些人会认为静坐或听音乐的时候思考得最好。还有一些会在阅读小说、学习不相关的学科、甚至在学习 Linux 系统的时候可以最佳思考。关键是我们都有不同的方式激发我们的创造力,而这些创造力的推进器中很多并不涉及键盘上的任何一个按键。我们真正的生产力对于系统管理员周围的人来说可能是完全看不见的。”
|
||||||
|
|
||||||
|
成为懒惰的系统管理员有一些简单的秘诀 —— 系统管理员要完成一切需要完成的事,而且更多的是,当所有人都处于恐慌的状态时要保持冷静和镇定。秘诀的一部分是高效工作,另一部分是把预防问题放在首位。
|
||||||
|
|
||||||
|
### 成为懒惰系统管理员的方法
|
||||||
|
|
||||||
|
#### 多思考
|
||||||
|
|
||||||
|
我相信关于懒惰系统管理员最重要的秘诀在于思考。正如上面的摘录所言,优秀的系统管理员会花大量的时候思考这些事情,如何更有效率地工作,在异常成为问题前如何定位,更聪明地工作,做其它事情的同时会考虑如何完成这些事情等等。
|
||||||
|
|
||||||
|
例如,除了撰写本文之外,我现在正在想一个项目,我打算在从亚马逊和本地计算机商店采购的新部件到达时才开始。我有一台不太关键的计算机上的主板坏了,最近它的崩溃更频率。但我的一台非常老的小服务器并没有出现故障,它负责处理我的电子邮件和外部网站,以及为我的网络的其余部分提供 DHCP 和 DNS 服务,但需要解决由于各种外部攻击而导致的间歇性过载。
|
||||||
|
|
||||||
|
我一开始想,我只要替换故障设备的主板及其直接部件:内存、CPU,可能还有电源。但是在考虑了一段时间之后,我决定将新部件放到服务器中,并将旧的(但仍然可用的)部件从服务器移到故障设备中。可以这样做的,只需一、两个小时就可以从服务器上移除旧部件并安装新的。然后我就可以花时间更换出故障的电脑里的部件了。太好了,所以我开始在脑海中列出要完成的任务。
|
||||||
|
|
||||||
|
然而,当我查看这个任务列表时,我意识到服务器中唯一不能替换的部件是机箱和硬盘驱动器,这两台计算机的机箱几乎完全相同。在有了这个小小的发现之后,我开始考虑用新的部件替换出了故障的计算机的部件,并将之作为我的服务器。经过一些测试之后,我只需从当前的服务器移除硬盘,并将它安装到用了新组件的机箱中,改下网络配置项,再更改 KVM 交换机端口上的主机名,并更改机箱上的主机名标签,就可以了。这将大大减少服务器停机时间,大大减少我的压力。此外,如果出现故障,我可以简单地将硬盘移回原来的服务器,直到我可以用新服务器解决问题为止。
|
||||||
|
|
||||||
|
所以,现在我在脑海中已经创建了一个完成这项工作我所需要做的任务清单。而且,我希望你能仔细观察,当我脑子里想着这一切的时候,我的手指从来没有碰过键盘。我新的心理行动计划风险很低,与我最初的计划相比,涉及的服务器停机时间要少得多。
|
||||||
|
|
||||||
|
当我在 IBM 工作的时候,我经常看到很多语言中都有写着“**思考**”的标语。思考可以节省时间和压力,是懒散的系统管理员的主要标志。
|
||||||
|
|
||||||
|
#### 做预防性维护
|
||||||
|
|
||||||
|
在 1970 年代中期,我被 IBM 聘为客户工程师,我的领地由相当多的[穿孔卡片设备][2]组成。这也就是说,它们是处理打孔卡的重型机械设备,其中一些可以追溯到 20 世纪 30 年代。因为这些机器主要是机械的,所以它们的部件经常磨损或失调。我的部分工作是在它们损坏时修复它们。我工作的主要部分,也是最重要的部分,是首先要防止它们损坏。预防性维护的目的是在磨损部件损坏之前进行更换,并对还在运行的部件进行润滑和调整,以确保它们工作正常。
|
||||||
|
|
||||||
|
正如我在《系统管理员的 Linux 哲学》中所言:
|
||||||
|
|
||||||
|
>“我在 IBM 的经理们明白这只是冰山一角;他们和我都知道,我的工作是让顾客满意。虽然这通常意味着修复损坏的硬件,但也意味着减少硬件损坏的次数。这对客户来说是好事,因为他们的机器在工作时工作效率更高。这对我有好处,因为我从那些快乐的客户那里接到的电话要少得多。我也睡了更多的觉,因为这样做的结果是更少的非工作时间的紧急电话。我是个懒惰的(客户工程师)。通过提前做额外的工作,从长远来看,我需要做的工作要少得多。
|
||||||
|
>
|
||||||
|
>“这一原则已成为系统管理员的 Linux 哲学的功能原则之一。作为系统管理员,我们的时间最好用在最大限度地减少未来工作量的任务上。”
|
||||||
|
|
||||||
|
在 Linux 计算机中查找要解决的问题相当于项目管理。我检查系统日志,寻找以后可能会变得非常危险的问题的迹象。如果出现了一些小问题,或者我注意到我的工作站、服务器没有做出该有的响应,或者如果日志显示了一些不寻常的东西,所有这些都可以暗示出潜在的问题,而对于用户或半秃头老板来说,这些问题并没有产生明显的症状。
|
||||||
|
|
||||||
|
我经常检查 `/var/log/` 中的文件,特别是 `messages` 和 `security` 文件。我最常见的问题之一是许多脚本小子在我的防火墙系统上尝试各种类型的攻击。而且,不,我不依赖 ISP 提供的调制解调器/路由器中的所谓的防火墙。这些日志包含了大量关于企图攻击来源的信息,非常有价值。但是要扫描不同主机上的日志并将解决方案部署到位,需要做大量的工作,所以我转向自动化。
|
||||||
|
|
||||||
|
#### 自动化
|
||||||
|
|
||||||
|
我发现我的工作有很大一部分可以通过某种形式的自动化来完成。系统管理员的 Linux 哲学的原则之一是 “自动化一切”,这包括每天扫描日志文件等枯燥乏味的任务。
|
||||||
|
|
||||||
|
像是 [Logwatch][3] 这类的程序能够监控你的日志文件中的异常条目,并在异常条目发生时通知您。Logwatch 通常作为 cron 任务每天运行一次,并向本地主机上的 root 用户发送电子邮件。你可以从命令行运行 Logwatch,并立即在显示器上查看结果。现在我只需要每天查看 Logwatch 的电子邮件通知。
|
||||||
|
|
||||||
|
但现实是,仅仅收到通知是不够的,因为我们不能坐以待毙。有时需要立即作出反应。我喜欢的另一个程序是——它能为我做所有事(看,这就是懒惰的管理员)——它就是 [Fail2ban][4]。Fail2Ban 会扫描指定的日志文件,查找各种类型的黑客攻击和入侵尝试,如果它发现某个 IP 地址在持续做特定类型的活动,它会向防火墙添加一个条目,在指定的时间内阻止来自该 IP 地址的任何进一步的黑客尝试。默认值通常在 10 分钟左右,但我喜欢为大多数类型的攻击指定为 12 或 24 小时。每种类型的黑客攻击都是单独配置的,例如尝试通过 SSH 登录和那些 Web 服务器的攻击。
|
||||||
|
|
||||||
|
#### 写脚本
|
||||||
|
|
||||||
|
自动化是这种哲学的关键组成部分之一。一切可以自动化的东西都应该自动化的,其余的尽可能地自动化。所以,我也写了很多脚本来解决问题,也就是说我编写了脚本来完成我的大部分工作。
|
||||||
|
|
||||||
|
我的脚本帮我节省了大量时间,因为它们包含执行特定任务的命令,这大大减少了我需要输入的数量。例如,我经常重新启动我的电子邮件服务器和垃圾邮件过滤软件(当修改 SpamAssassin 的 `local.cf` 配置文件时,就需要重启)。必须按特定顺序停止并重新启动这些服务。因此,我用几个命令编写了一个简短的脚本,并将其存储在可访问的 `/usr/local/bin` 中。现在,不用键入几个命令并等待每个命令都完成,然后再键入下一个命令,更不用记住正确的命令顺序和每个命令的正确语法,我输入一个三个字符的命令,其余的留给我的脚本来完成。
|
||||||
|
|
||||||
|
#### 简化键入
|
||||||
|
|
||||||
|
另一种成为懒惰的系统管理员的方法是减少我们需要键入的数量。而且,我的打字技巧真的很糟糕(也就是说,我一点也没有,顶多是几个笨拙的手指)。导致错误的一个可能原因是我糟糕的打字技巧,所以我会尽量少打字。
|
||||||
|
|
||||||
|
绝大多数 GNU 和 Linux 核心实用程序都有非常短的名称。然而,它们都是有意义的名字。诸如用于更改目录的 `cd` 、用于列出目录内容的 `ls` 和用于磁盘转储的 `dd` 等工具都一目了然。短名字意味着更少的打字和更少的产生错误机会。我认为短的名字通常更容易记住。
|
||||||
|
|
||||||
|
当我编写 shell 脚本时,我喜欢保持名称简短而意义(至少对我来说是),比如用于 rsync 备份的 `rsbu`(LCTT 译注,Rsync Backup 的简写)。但在某些情况下,我喜欢使用更长的名称,比如 `doUpdates` 来执行系统更新。在后一种情况下,更长一点的名字让脚本的目的更明显。这可以节省时间,因为很容易记住脚本的名称。
|
||||||
|
|
||||||
|
其他减少键入的方法包括命令行别名、历史命令调回和编辑。别名只是你在 Bash shell 键入命令时才做的替换。键入 `alias` 命令会看到默认配置的别名列表。例如,当你输入命令 `ls` 时,会被条目 `alias ls='ls –color=auto'` 替成较长的命令,因此你只需键入 2 个字符而不是 14 个字符即可获得带有颜色的文件列表。还可以使用 `alias` 命令添加你自己定义的别名。
|
||||||
|
|
||||||
|
历史命令调回允许你使用键盘的向上和向下箭头键滚动浏览命令历史记录。如果需要再次使用相同的命令,只需在找到所需的命令时按回车键即可。如果在找到命令后需要更改该命令,则可以使用标准命令行编辑功能进行更改。
|
||||||
|
|
||||||
|
### 结束语
|
||||||
|
|
||||||
|
一名懒惰的系统管理员实际上也有很多的工作。但我们是聪明地工作,而不是刻苦工作。早在一堆小问题汇聚成大问题之前,我们就花时间探索我们负责的主机,并处理好所有的小问题。我们花了很多时间思考解决问题的最佳方法,我们也花了很多时间来发现新的方法,让自己更聪明地工作,成为懒惰的系统管理员。
|
||||||
|
|
||||||
|
除了这里描述的少数方法外,还有许多其他的方式可以成为懒惰的系统管理员。我相信你也有一些自己的方式;请在评论中和我们分享。
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: https://opensource.com/article/18/7/how-be-lazy-sysadmin
|
||||||
|
|
||||||
|
作者:[David Both][a]
|
||||||
|
选题:[lujun9972](https://github.com/lujun9972)
|
||||||
|
译者:[zgj1024](https://github.com/zgj1024)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]:https://opensource.com/users/dboth
|
||||||
|
[1]:https://www.apress.com/us/book/9781484237298
|
||||||
|
[2]:https://en.wikipedia.org/wiki/Unit_record_equipment
|
||||||
|
[3]:https://www.techrepublic.com/article/how-to-install-and-use-logwatch-on-linux/
|
||||||
|
[4]:https://www.fail2ban.org/wiki/index.php/Main_Page
|
@ -0,0 +1,128 @@
|
|||||||
|
dbxfs:在 Linux 中本地挂载 Dropbox 文件夹
|
||||||
|
======
|
||||||
|
|
||||||
|
![](https://www.ostechnix.com/wp-content/uploads/2018/10/dbxfs-720x340.png)
|
||||||
|
|
||||||
|
不久前,我们总结了所有 [在本地挂载 Google Drive][1] 作为虚拟文件系统,并从 Linux 系统访问存储在 Google Drive 中的文件的方法。今天,我们将学习使用 `dbxfs` 将 Dropbox 文件夹挂载到本地文件系统中。`dbxfs` 用于在类 Unix 操作系统中本地挂载 Dropbox 文件夹作为虚拟文件系统。虽然在 Linux 中很容易[安装 Dropbox 客户端][2],但这种方法与官方方法略有不同。它是一个命令行 dropbox 客户端,且无需磁盘空间即可访问。`dbxfs` 是自由开源的,并且是用 Python 3.5+ 编写的。
|
||||||
|
|
||||||
|
### 安装 dbxfs
|
||||||
|
|
||||||
|
`dbxfs` 官方支持 Linux 和 Mac OS。但是,它应该适用于任何提供 **FUSE 兼容库**或能够挂载 SMB 共享的 POSIX 系统。由于它是用 Python 3.5 编写的,因此可以使用 pip3 包管理器进行安装。如果尚未安装 pip,请参阅以下指南。
|
||||||
|
|
||||||
|
- [如何使用 pip 管理 Python 包][13]
|
||||||
|
|
||||||
|
并且也要安装 FUSE 库。
|
||||||
|
|
||||||
|
在基于 Debian 的系统上,运行以下命令以安装 FUSE:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ sudo apt install libfuse2
|
||||||
|
```
|
||||||
|
|
||||||
|
在 Fedora 上:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ sudo dnf install fuse
|
||||||
|
```
|
||||||
|
|
||||||
|
安装完所有必需的依赖项后,运行以下命令以安装 `dbxfs`:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ pip3 install dbxfs
|
||||||
|
```
|
||||||
|
|
||||||
|
### 在本地挂载 Dropbox 文件夹
|
||||||
|
|
||||||
|
创建一个挂载点以将 Dropbox 文件夹挂载到本地文件系统中。
|
||||||
|
|
||||||
|
```
|
||||||
|
$ mkdir ~/mydropbox
|
||||||
|
```
|
||||||
|
|
||||||
|
然后,使用 `dbxfs` 在本地挂载 dropbox 文件夹,如下所示:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ dbxfs ~/mydropbox
|
||||||
|
```
|
||||||
|
|
||||||
|
你将被要求生成一个访问令牌:
|
||||||
|
|
||||||
|
![](https://www.ostechnix.com/wp-content/uploads/2018/10/Generate-access-token-1.png)
|
||||||
|
|
||||||
|
要生成访问令牌,只需在 Web 浏览器中输入上面输出的 URL,然后单击 **允许** 以授权 Dropbox 访问。你需要登录 Dropbox 帐户才能完成授权过程。
|
||||||
|
|
||||||
|
下一个页面将生成新的授权码。复制代码并返回终端将其粘贴到 cli-dbxfs 提示符中以完成该过程。
|
||||||
|
|
||||||
|
然后,系统会要求你保存凭据以供将来访问。根据你是要保存还是拒绝,输入 `Y` 或 `N`。然后,你需要为新的访问令牌输入两次密码。
|
||||||
|
|
||||||
|
最后,输入 `Y` 接受 `/home/username/mydropbox` 作为默认挂载点。如果你要设置不同的路径,输入 `N` 并输入你选择的位置。
|
||||||
|
|
||||||
|
![Generate access token 2][4]
|
||||||
|
|
||||||
|
完成了!从现在开始,你可以看到你的 Dropbox 文件夹已挂载到本地文件系统中。
|
||||||
|
|
||||||
|
![](https://www.ostechnix.com/wp-content/uploads/2018/10/Dropbox-in-file-manager.png)
|
||||||
|
|
||||||
|
### 更改访问令牌存储路径
|
||||||
|
|
||||||
|
默认情况下,`dbxfs` 会将 Dropbox 访问令牌存储在系统密钥环或加密文件中。但是,你可能希望将其存储在 gpg 加密文件或其他地方。如果是这样,请在 [Dropbox 开发者应用控制台][5]上创建个人应用来获取访问令牌。
|
||||||
|
|
||||||
|
![](https://www.ostechnix.com/wp-content/uploads/2018/10/access-token.png)
|
||||||
|
|
||||||
|
创建应用后,单击下一步中的**生成**按钮。此令牌可用于通过 API 访问你的 Dropbox 帐户。不要与任何人共享你的访问令牌。
|
||||||
|
|
||||||
|
![](https://www.ostechnix.com/wp-content/uploads/2018/10/Create-a-new-app.png)
|
||||||
|
|
||||||
|
创建访问令牌后,使用任何你选择的加密工具对其进行加密,例如 [Cryptomater][6]、[Cryptkeeper][7]、[CryptGo][8]、[Cryptr][9]、[Tomb][10]、[Toplip][11] 和 [**GnuPG][12] 等,并在你喜欢的位置保存。
|
||||||
|
|
||||||
|
接下来编辑 dbxfs 配置文件并添加以下行:
|
||||||
|
|
||||||
|
```
|
||||||
|
"access_token_command": ["gpg", "--decrypt", "/path/to/access/token/file.gpg"]
|
||||||
|
```
|
||||||
|
|
||||||
|
你可以通过运行以下命令找到 dbxfs 配置文件:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ dbxfs --print-default-config-file
|
||||||
|
```
|
||||||
|
|
||||||
|
有关更多详细信息,请参阅 dbxfs 帮助:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ dbxfs -h
|
||||||
|
```
|
||||||
|
|
||||||
|
如你所见,使用 `dbxfs` 在你的文件系统中本地挂载 Dropfox 文件夹并不复杂。经过测试,`dbxfs` 如常工作。如果你有兴趣了解它是如何工作的,请尝试一下,并在下面的评论栏告诉我们你的体验。
|
||||||
|
|
||||||
|
就是这些了。希望这篇文章有用。还有更多好东西。敬请期待!
|
||||||
|
|
||||||
|
干杯!
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: https://www.ostechnix.com/dbxfs-mount-dropbox-folder-locally-as-virtual-file-system-in-linux/
|
||||||
|
|
||||||
|
作者:[SK][a]
|
||||||
|
选题:[lujun9972](https://github.com/lujun9972)
|
||||||
|
译者:[geekpi](https://github.com/geekpi)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]: https://www.ostechnix.com/author/sk/
|
||||||
|
[1]: https://www.ostechnix.com/how-to-mount-google-drive-locally-as-virtual-file-system-in-linux/
|
||||||
|
[2]: https://www.ostechnix.com/install-dropbox-in-ubuntu-18-04-lts-desktop/
|
||||||
|
[3]: 
|
||||||
|
[4]: http://www.ostechnix.com/wp-content/uploads/2018/10/Generate-access-token-2.png
|
||||||
|
[5]: https://dropbox.com/developers/apps
|
||||||
|
[6]: https://www.ostechnix.com/cryptomator-open-source-client-side-encryption-tool-cloud/
|
||||||
|
[7]: https://www.ostechnix.com/how-to-encrypt-your-personal-foldersdirectories-in-linux-mint-ubuntu-distros/
|
||||||
|
[8]: https://www.ostechnix.com/cryptogo-easy-way-encrypt-password-protect-files/
|
||||||
|
[9]: https://www.ostechnix.com/cryptr-simple-cli-utility-encrypt-decrypt-files/
|
||||||
|
[10]: https://www.ostechnix.com/tomb-file-encryption-tool-protect-secret-files-linux/
|
||||||
|
[11]: https://www.ostechnix.com/toplip-strong-file-encryption-decryption-cli-utility/
|
||||||
|
[12]: https://www.ostechnix.com/an-easy-way-to-encrypt-and-decrypt-files-from-commandline-in-linux/
|
||||||
|
[13]:https://www.ostechnix.com/manage-python-packages-using-pip/
|
@ -0,0 +1,111 @@
|
|||||||
|
怎样如软件工程师一样组织知识
|
||||||
|
==========
|
||||||
|
|
||||||
|
总体上说,软件开发和技术是以非常快的速度发展的领域,所以持续学习是必不可少的。在互联网上花几分钟找一下,在 Twitter、媒体、RSS 订阅、Hacker News 和其它专业网站和社区等地方,就可以从文章、案例研究、教程、代码片段、新应用程序和信息中找到大量有用的信息。
|
||||||
|
|
||||||
|
保存和组织所有这些信息可能是一项艰巨的任务。在这篇文章中,我将介绍一些我用来组织信息的工具。
|
||||||
|
|
||||||
|
我认为在知识管理方面非常重要的一点就是避免锁定在特定平台。我使用的所有工具都允许以标准格式(如 Markdown 和 HTML)导出数据。
|
||||||
|
|
||||||
|
请注意,我的流程并不完美,我一直在寻找新工具和方法来优化它。每个人都不同,所以对我有用的东西可能不适合你。
|
||||||
|
|
||||||
|
|
||||||
|
### 用 NotionHQ 做知识库
|
||||||
|
|
||||||
|
对我来说,知识管理的基本部分是拥有某种个人知识库或维基。这是一个你可以以有组织的方式保存链接、书签、备注等的地方。
|
||||||
|
|
||||||
|
我使用 [NotionHQ][7] 做这件事。我使用它来记录各种主题,包括资源列表,如通过编程语言分组的优秀的库或教程,为有趣的博客文章和教程添加书签等等,不仅与软件开发有关,而且与我的个人生活有关。
|
||||||
|
|
||||||
|
我真正喜欢 NotionHQ 的是,创建新内容是如此简单。你可以使用 Markdown 编写它并将其组织为树状。
|
||||||
|
|
||||||
|
这是我的“开发”工作区的顶级页面:
|
||||||
|
|
||||||
|
[![Image](https://res.cloudinary.com/practicaldev/image/fetch/s--uMbaRUtu--/c_limit%2Cf_auto%2Cfl_progressive%2Cq_auto%2Cw_880/http://i.imgur.com/kRnuvMV.png)][8]
|
||||||
|
|
||||||
|
NotionHQ 有一些很棒的其他功能,如集成了电子表格/数据库和任务板。
|
||||||
|
|
||||||
|
如果您想认真使用 NotionHQ,您将需要订阅付费个人计划,因为免费计划有所限制。我觉得它物有所值。NotionHQ 允许将整个工作区导出为 Markdown 文件。导出功能存在一些重要问题,例如丢失页面层次结构,希望 Notion 团队可以改进这一点。
|
||||||
|
|
||||||
|
作为一个免费的替代方案,我可能会使用 [VuePress][9] 或 [GitBook][10] 来托管我自己的知识库。
|
||||||
|
|
||||||
|
### 用 Pocket 保存感兴趣的文章
|
||||||
|
|
||||||
|
[Pocket][11] 是我最喜欢的应用之一!使用 Pocket,您可以创建一个来自互联网上的文章的阅读列表。每当我看到一篇看起来很有趣的文章时,我都会使用 Chrome 扩展程序将其保存到 Pocket。稍后,我会阅读它,如果我发现它足够有用,我将使用 Pocket 的“存档”功能永久保存该文章并清理我的 Pocket 收件箱。
|
||||||
|
|
||||||
|
我尽量保持这个阅读清单足够小,并存档我已经处理过的信息。Pocket 允许您标记文章,以便以后更轻松地搜索特定主题的文章。
|
||||||
|
|
||||||
|
如果原始网站消失,您还可以在 Pocket 服务器中保存文章的副本,但是您需要 Pocket Premium 订阅计划。
|
||||||
|
|
||||||
|
Pocket 还具有“发现”功能,根据您保存的文章推荐类似的文章。这是找到可以阅读的新内容的好方法。
|
||||||
|
|
||||||
|
### 用 SnippetStore 做代码片段管理
|
||||||
|
|
||||||
|
从 GitHub 到 Stack Overflow 的答案,到博客文章,经常能找到一些你想要保存备用的好代码片段。它可能是一些不错的算法实现、一个有用的脚本或如何在某种语言中执行某种操作的示例。
|
||||||
|
|
||||||
|
我尝试了很多应用程序,从简单的 GitHub Gists 到 [Boostnote][12],直到我发现 [SnippetStore][13]。
|
||||||
|
|
||||||
|
SnippetStore 是一个开源的代码片段管理应用。SnippetStore 与其他产品的区别在于其简单性。您可以按语言或标签整理片段,并且可以拥有多个文件片段。它不完美,但是可以用。例如,Boostnote 具有更多功能,但我更喜欢 SnippetStore 组织内容的简单方法。
|
||||||
|
|
||||||
|
对于我每天使用的缩写和片段,我更喜欢使用我的编辑器 / IDE 的代码片段功能,因为它更便于使用。我使用 SnippetStore 更像是作为编码示例的参考。
|
||||||
|
|
||||||
|
[Cacher][14] 也是一个有趣的选择,因为它与许多编辑器进行了集成,他有一个命令行工具,并使用 GitHub Gists 作为后端,但其专业计划为 6 美元/月,我觉这有点太贵。
|
||||||
|
|
||||||
|
### 用 DevHints 管理速查表
|
||||||
|
|
||||||
|
[Devhints][15] 是由 Rico Sta. Cruz 创建的一个速查表集合。它是开源的,是用 Jekyll 生成的,Jekyll 是最受欢迎的静态站点生成器之一。
|
||||||
|
|
||||||
|
这些速查表是用 Markdown 编写的,带有一些额外的格式化支持,例如支持列。
|
||||||
|
|
||||||
|
我非常喜欢其界面的外观,并且不像可以在 [Cheatography][16] 等网站上找到 PDF 或图像格式的速查表, Markdown 非常容易添加新内容并保持更新和进行版本控制。
|
||||||
|
|
||||||
|
因为它是开源,我创建了自己的分叉版本,删除了一些我不需要的速查表,并添加了更多。
|
||||||
|
|
||||||
|
我使用速查表作为如何使用某些库或编程语言或记住一些命令的参考。速查表的单个页面非常方便,例如,可以列出特定编程语言的所有基本语法。
|
||||||
|
|
||||||
|
我仍在尝试这个工具,但到目前为止它的工作很好。
|
||||||
|
|
||||||
|
### Diigo
|
||||||
|
|
||||||
|
[Diigo][17] 允许您注释和突出显示部分网站。我在研究新东西时使用它来注释重要信息,或者从文章、Stack Overflow 答案或来自 Twitter 的鼓舞人心的引语中保存特定段落!;)
|
||||||
|
|
||||||
|
* * *
|
||||||
|
|
||||||
|
就这些了。某些工具的功能方面可能存在一些重叠,但正如我在开始时所说的那样,这是一个不断演进的工作流程,因为我一直在尝试和寻找改进和提高工作效率的方法。
|
||||||
|
|
||||||
|
你呢?是如何组织你的知识的?请随时在下面发表评论。
|
||||||
|
|
||||||
|
谢谢你的阅读。
|
||||||
|
|
||||||
|
------------------------------------------------------------------------
|
||||||
|
|
||||||
|
作者简介:Bruno Paz,Web 工程师,专精 #PHP 和 @Symfony 框架。热心于新技术。喜欢运动,@FCPorto 的粉丝!
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: https://dev.to/brpaz/how-do-i-organize-my-knowledge-as-a-software-engineer-4387
|
||||||
|
|
||||||
|
作者:[Bruno Paz][a]
|
||||||
|
选题:[oska874](https://github.com/oska874)
|
||||||
|
译者:[wxy](https://github.com/wxy)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]:http://brunopaz.net/
|
||||||
|
[1]:https://dev.to/brpaz
|
||||||
|
[2]:http://twitter.com/brunopaz88
|
||||||
|
[3]:http://github.com/brpaz
|
||||||
|
[4]:https://dev.to/t/knowledge
|
||||||
|
[5]:https://dev.to/t/learning
|
||||||
|
[6]:https://dev.to/t/development
|
||||||
|
[7]:https://www.notion.so/
|
||||||
|
[8]:https://res.cloudinary.com/practicaldev/image/fetch/s--uMbaRUtu--/c_limit%2Cf_auto%2Cfl_progressive%2Cq_auto%2Cw_880/http://i.imgur.com/kRnuvMV.png
|
||||||
|
[9]:https://vuepress.vuejs.org/
|
||||||
|
[10]:https://www.gitbook.com/?t=1
|
||||||
|
[11]:https://getpocket.com/
|
||||||
|
[12]:https://boostnote.io/
|
||||||
|
[13]:https://github.com/ZeroX-DG/SnippetStore
|
||||||
|
[14]:https://www.cacher.io/
|
||||||
|
[15]:https://devhints.io/
|
||||||
|
[16]:https://cheatography.com/
|
||||||
|
[17]:https://www.diigo.com/index
|
188
published/201812/20171012 7 Best eBook Readers for Linux.md
Normal file
188
published/201812/20171012 7 Best eBook Readers for Linux.md
Normal file
@ -0,0 +1,188 @@
|
|||||||
|
7 个最佳 Linux 电子书阅读器
|
||||||
|
======
|
||||||
|
|
||||||
|
**摘要:** 本文中我们涉及一些 Linux 最佳电子书阅读器。这些应用提供更佳的阅读体验甚至可以管理你的电子书。
|
||||||
|
|
||||||
|
![最佳 Linux 电子书阅读器][1]
|
||||||
|
|
||||||
|
最近,随着人们发现在手持设备、Kindle 或者 PC 上阅读更加舒适,对电子图书的需求有所增加。至于 Linux 用户,也有各种电子书应用满足你阅读和整理电子书的需求。
|
||||||
|
|
||||||
|
在本文中,我们选出了七个最佳 Linux 电子书阅读器。这些电子书阅读器最适合 pdf、epub 和其他电子书格式。
|
||||||
|
|
||||||
|
我提供的是 Ubuntu 安装说明,因为我现在使用它。如果你使用的是[非 Ubuntu 发行版][2],你能在你的发行版软件仓库中找到大多数这些电子书应用。
|
||||||
|
|
||||||
|
### 1. Calibre
|
||||||
|
|
||||||
|
[Calibre][3] 是 Linux 最受欢迎的电子书应用。老实说,这不仅仅是一个简单的电子书阅读器。它是一个完整的电子书解决方案。你甚至能[通过 Calibre 创建专业的电子书][4]。
|
||||||
|
|
||||||
|
通过强大的电子书管理和易用的界面,它提供了创建和编辑电子书的功能。Calibre 支持多种格式和与其它电子书阅读器同步。它也可以让你轻松转换一种电子书格式到另一种。
|
||||||
|
|
||||||
|
Calibre 最大的缺点是,资源消耗太多,因此作为一个独立的电子阅读器来说是一个艰难的选择。
|
||||||
|
|
||||||
|
![Calibre][5]
|
||||||
|
|
||||||
|
#### 特性
|
||||||
|
|
||||||
|
* 管理电子书:Calibre 通过管理元数据来排序和分组电子书。你能从各种来源下载一本电子书的元数据或创建和编辑现有的字段。
|
||||||
|
* 支持所有主流电子书格式:Calibre 支持所有主流电子书格式并兼容多种电子阅读器。
|
||||||
|
* 文件转换:在转换时,你能通过改变电子书风格,创建内容表和调整边距的选项来转换任何一种电子书格式到另一种。你也能转换个人文档为电子书。
|
||||||
|
* 从 web 下载杂志期刊:Calibre 能从各种新闻源或者通过 RSS 订阅源传递故事。
|
||||||
|
* 分享和备份你的电子图书馆:它提供了一个选项,可以托管你电子书集合到它的服务端,从而你能与好友共享或用任何设备从任何地方访问。备份和导入/导出特性可以确保你的收藏安全和方便携带。
|
||||||
|
|
||||||
|
#### 安装
|
||||||
|
|
||||||
|
你能在主流 Linux 发行版的软件库中找到它。对于 Ubuntu,在软件中心搜索它或者使用下面的命令:
|
||||||
|
|
||||||
|
```
|
||||||
|
sudo apt-get install calibre
|
||||||
|
```
|
||||||
|
|
||||||
|
### 2. FBReader
|
||||||
|
|
||||||
|
![FBReader: Linux 电子书阅读器][6]
|
||||||
|
|
||||||
|
[FBReader][7] 是一个开源的轻量级多平台电子书阅读器,它支持多种格式,比如 ePub、fb2、mobi、rtf、html 等。它包括了一些可以访问的流行网络电子图书馆,那里你能免费或付费下载电子书。
|
||||||
|
|
||||||
|
#### 特性
|
||||||
|
|
||||||
|
* 支持多种文件格式和设备比如 Android、iOS、Windows、Mac 和更多。
|
||||||
|
* 同步书集、阅读位置和书签。
|
||||||
|
* 在线管理你图书馆,可以从你的 Linux 桌面添加任何书到所有设备。
|
||||||
|
* 支持 Web 浏览器访问你的书集。
|
||||||
|
* 支持将书籍存储在 Google Drive ,可以通过作者,系列或其他属性整理书籍。
|
||||||
|
|
||||||
|
#### 安装
|
||||||
|
|
||||||
|
你能从官方库或者在终端中输入以下命令安装 FBReader 电子阅读器。
|
||||||
|
|
||||||
|
```
|
||||||
|
sudo apt-get install fbreader
|
||||||
|
```
|
||||||
|
|
||||||
|
或者你能从[这里][8]抓取一个以 .deb 包,并在你的基于 Debian 发行版的系统上安装它。
|
||||||
|
|
||||||
|
### 3. Okular
|
||||||
|
|
||||||
|
[Okular][9] 是另一个开源的基于 KDE 开发的跨平台文档查看器,它已经作为 KDE 应用发布的一部分了。
|
||||||
|
|
||||||
|
![Okular][10]
|
||||||
|
|
||||||
|
#### 特性
|
||||||
|
|
||||||
|
* Okular 支持多种文档格式像 PDF、Postscript、DjVu、CHM、XPS、ePub 和其他。
|
||||||
|
* 支持在 PDF 文档中评论、高亮和绘制不同的形状等。
|
||||||
|
* 无需修改原始 PDF 文件,分别保存上述这些更改。
|
||||||
|
* 电子书中的文本能被提取到一个文本文件,并且有个名为 Jovie 的内置文本阅读服务。
|
||||||
|
|
||||||
|
备注:查看这个应用的时候,我发现这个应用在 Ubuntu 和它的衍生系统中不支持 ePub 文件格式。其他发行版用户仍然可以发挥它全部的潜力。
|
||||||
|
|
||||||
|
#### 安装
|
||||||
|
|
||||||
|
Ubuntu 用户可以在终端中键入下面的命令来安装它:
|
||||||
|
|
||||||
|
```
|
||||||
|
sudo apt-get install okular
|
||||||
|
```
|
||||||
|
|
||||||
|
### 4. Lucidor
|
||||||
|
|
||||||
|
Lucidor 是一个易用的、支持 epub 文件格式和在 OPDS 格式中编目的电子阅读器。它也具有在本地书架里组织电子书集、从互联网搜索和下载,和将 Web 订阅和网页转换成电子书的功能。
|
||||||
|
|
||||||
|
Lucidor 是 XULRunner 应用程序,它向您展示了具有类似火狐的选项卡式布局,和存储数据和配置时的行为。它是这个列表中最简单的电子阅读器,包括诸如文本说明和滚动选项之类的配置。
|
||||||
|
|
||||||
|
![lucidor][11]
|
||||||
|
|
||||||
|
你可以通过选择单词并右击“查找单词”来查找该单词在 Wiktionary.org 的定义。它也包含 web 订阅或 web 页面作为电子书的选项。
|
||||||
|
|
||||||
|
你能从[这里][12]下载和安装 deb 或者 RPM 包。
|
||||||
|
|
||||||
|
### 5. Bookworm
|
||||||
|
|
||||||
|
![Bookworm Linux 电子阅读器][13]
|
||||||
|
|
||||||
|
Bookworm 是另一个支持多种文件格式诸如 epub、pdf、mobi、cbr 和 cbz 的自由开源的电子阅读器。我写了一篇关于 Bookworm 应用程序的特性和安装的专题文章,到这里阅读:[Bookworm:一个简单而强大的 Linux 电子阅读器][14]
|
||||||
|
|
||||||
|
#### 安装
|
||||||
|
|
||||||
|
```
|
||||||
|
sudo apt-add-repository ppa:bookworm-team/bookworm
|
||||||
|
sudo apt-get update
|
||||||
|
sudo apt-get install bookworm
|
||||||
|
```
|
||||||
|
|
||||||
|
### 6. Easy Ebook Viewer
|
||||||
|
|
||||||
|
[Easy Ebook Viewer][15] 是又一个用于读取 ePub 文件的很棒的 GTK Python 应用。具有基本章节导航、从上次阅读位置继续、从其他电子书文件格式导入、章节跳转等功能,Easy Ebook Viewer 是一个简单而简约的 ePub 阅读器.
|
||||||
|
|
||||||
|
![Easy-Ebook-Viewer][16]
|
||||||
|
|
||||||
|
这个应用仍然处于初始阶段,只支持 ePub 文件。
|
||||||
|
|
||||||
|
#### 安装
|
||||||
|
|
||||||
|
你可以从 [GitHub][17] 下载源代码,并自己编译它及依赖项来安装 Easy Ebook Viewer。或者,以下终端命令将执行完全相同的工作。
|
||||||
|
|
||||||
|
```
|
||||||
|
sudo apt install git gir1.2-webkit-3.0 libwebkitgtk-3.0-0 gir1.2-gtk-3.0 python3-gi
|
||||||
|
git clone https://github.com/michaldaniel/Ebook-Viewer.git
|
||||||
|
cd Ebook-Viewer/
|
||||||
|
sudo make install
|
||||||
|
```
|
||||||
|
|
||||||
|
成功完成上述步骤后,你可以从 Dash 启动它。
|
||||||
|
|
||||||
|
### 7. Buka
|
||||||
|
|
||||||
|
Buka 主要是一个具有简单而清爽的用户界面的电子书管理器。它目前支持 PDF 格式,旨在帮助用户更加关注内容。拥有 PDF 阅读器的所有基本特性,Buka 允许你通过箭头键导航,具有缩放选项,并且能并排查看两页。
|
||||||
|
|
||||||
|
你可以创建单独的 PDF 文件列表并轻松地在它们之间切换。Buka 也提供了一个内置翻译工具,但是你需要有效的互联网连接来使用这个特性。
|
||||||
|
|
||||||
|
![Buka][19]
|
||||||
|
|
||||||
|
#### 安装
|
||||||
|
|
||||||
|
你能从[官方下载页面][20]下载一个 AppImage。如果你不知道如何做,请阅读[如何在 Linux 下使用 AppImage][21]。或者,你可以通过命令行安装它:
|
||||||
|
|
||||||
|
```
|
||||||
|
sudo snap install buka
|
||||||
|
```
|
||||||
|
|
||||||
|
### 结束语
|
||||||
|
|
||||||
|
就我个人而言,我发现 Calibre 最适合我的需要。当然,Bookworm 看起来很有前途,这几天我经常使用它。不过,电子书应用的选择完全取决于你的喜好。
|
||||||
|
|
||||||
|
你使用哪个电子书应用呢?在下面的评论中让我们知道。
|
||||||
|
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: https://itsfoss.com/best-ebook-readers-linux/
|
||||||
|
|
||||||
|
作者:[Ambarish Kumar][a]
|
||||||
|
译者:[zjon](https://github.com/zjon)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]:https://itsfoss.com/author/ambarish/
|
||||||
|
[1]:https://i0.wp.com/itsfoss.com/wp-content/uploads/2017/10/best-ebook-readers-linux.png
|
||||||
|
[2]:https://itsfoss.com/non-ubuntu-beginner-linux/
|
||||||
|
[3]:https://www.calibre-ebook.com
|
||||||
|
[4]:https://itsfoss.com/create-ebook-calibre-linux/
|
||||||
|
[5]:https://i0.wp.com/itsfoss.com/wp-content/uploads/2017/09/Calibre-800x603.jpeg
|
||||||
|
[6]:https://i0.wp.com/itsfoss.com/wp-content/uploads/2017/10/fbreader-800x624.jpeg
|
||||||
|
[7]:https://fbreader.org
|
||||||
|
[8]:https://fbreader.org/content/fbreader-beta-linux-desktop
|
||||||
|
[9]:https://okular.kde.org/
|
||||||
|
[10]:https://i0.wp.com/itsfoss.com/wp-content/uploads/2017/09/Okular-800x435.jpg
|
||||||
|
[11]:https://i0.wp.com/itsfoss.com/wp-content/uploads/2017/09/lucidor-2.png
|
||||||
|
[12]:http://lucidor.org/lucidor/download.php
|
||||||
|
[13]:https://i0.wp.com/itsfoss.com/wp-content/uploads/2017/08/bookworm-ebook-reader-linux-800x450.jpeg
|
||||||
|
[14]:https://itsfoss.com/bookworm-ebook-reader-linux/
|
||||||
|
[15]:https://github.com/michaldaniel/Ebook-Viewer
|
||||||
|
[16]:https://i0.wp.com/itsfoss.com/wp-content/uploads/2017/09/Easy-Ebook-Viewer.jpg
|
||||||
|
[17]:https://github.com/michaldaniel/Ebook-Viewer.git
|
||||||
|
[18]:https://github.com/oguzhaninan/Buka
|
||||||
|
[19]:https://i0.wp.com/itsfoss.com/wp-content/uploads/2017/09/Buka2-800x555.png
|
||||||
|
[20]:https://github.com/oguzhaninan/Buka/releases
|
||||||
|
[21]:https://itsfoss.com/use-appimage-linux/
|
286
published/201812/20171111 A CEOs Guide to Emacs.md
Normal file
286
published/201812/20171111 A CEOs Guide to Emacs.md
Normal file
@ -0,0 +1,286 @@
|
|||||||
|
CEO 的 Emacs 秘籍
|
||||||
|
===========
|
||||||
|
|
||||||
|
几年前,不,是几十年前,我就在用 Emacs。不论是码代码、编写文档,还是管理邮件和日程,我都用这个编辑器,或者是说操作系统,而且我还乐此不疲。许多年过去了,我也转向了其他更新、更好的工具。结果,就连最基本的文件浏览,我都已经忘了在不用鼠标的情况下该怎么操作。大约三个月前,我意识到我在应用程序和计算机之间切换上耗费了大量的时间,于是我决定再次使用 Emacs。这是个很正确的决定,原因有以下几个。其中包括用 `.emacs` 和 Dropbox 来搭建一个良好的、可移植的环境的一些技巧。
|
||||||
|
|
||||||
|
对于那些还没用过 Emacs 的人来说,Emacs 会让你爱恨交加。它有点像一个房子大小的<ruby>鲁布·戈德堡机械<rt>Rube Goldberg machine</rt></ruby>,乍一看,它具备烤面包机的所有功能。这听起来不像是一种认可,但关键词是“乍一看”。一旦你了解了 Emacs,你就会意识到它其实是一台可以当发动机用的热核烤面包机……好吧,只是指文本处理的所有事情。当考虑到你计算机的使用周期在很大程度上都是与文本有关时,这是一个相当大胆的声明。大胆,但却是真的。
|
||||||
|
|
||||||
|
也许对我来说更重要的是,Emacs 是我曾经使用过的一个应用,并让我觉得我真正的拥有它,而不是把我塑造成一个匿名的“用户”,就好像位于 [Soma][30](LCTT 译注:旧金山的一个街区)或雷蒙德(LCTT 译注:微软总部所在地)附近某个高档办公室的产品营销部门把钱作为明确的目标一样。现代生产力和创作应用程序(如 Pages 或 IDE)就像碳纤维赛车,它们装备得很好,也很齐全。而 Emacs 就像一盒经典的 [Campagnolo][31] (LCTT 译注:世界上最好的三个公路自行车套件系统品牌之一)零件和一个漂亮的自行车牵引式钢框架,但缺少曲柄臂和刹车杆,你必须在网上某个小众文化中找到它们。前者更快而且很完整,后者是无尽的快乐或烦恼的源泉,当然这取决于你自己,而且这种快乐或烦恼会伴随到你死。我就是那种在找到一堆老古董或用 `Emacs Lisp` 配置编辑器时会感到高兴的人,具体情况因人而异。
|
||||||
|
|
||||||
|
![1933 steel bicycle](https://www.fugue.co/hubfs/Imported_Blog_Media/bicycle-1.jpg)
|
||||||
|
|
||||||
|
*一辆我还在骑的 1933 年产的钢制自行车。你可以看看框架管差别: [https://www.youtube.com/watch?v=khJQgRLKMU0][6]*
|
||||||
|
|
||||||
|
这可能给人一种 Emacs 已经过气或过时的印象。然而并不是,Emacs 是强大和永恒的,只要你耐心地去理解它的一些规则。Emacs 的规则很另类,也很奇怪,但其中的逻辑却引人注目,且魅力十足。对于我来说, Emacs 更像是未来而不是过去。就像牵引式钢框架在未来几十年里将会变得好用和舒适,而神奇的碳纤维自行车将会被扔进垃圾场,在撞击中粉碎一样,Emacs 也将会作为一种在最新的流行应用早已被遗忘的时候的好用的工具继续存在这里。
|
||||||
|
|
||||||
|
使用 Lisp 代码来构建个人工作环境,并将这个恰到好处的环境移植到任何计算机,如果这种想法打动了你,那么你将会爱上 Emacs。如果你喜欢很潮、很炫的,又不想投入太多时间和精力的情况下就能直接工作的话,那么 Emacs 可能不适合你。我已经不再写代码了(除了 Ludwig 和 Emacs Lisp),但是 Fugue 公司的很多工程师都使用 Emacs 来提高码代码的效率。我公司有 30% 的工程师用 Emacs,40% 用 IDE 和 30% 的用 vim。但这篇文章是写给 CEO 和其他<ruby>[精英][32]<rt>Pointy-Haired Bosses</rt></ruby>(PHB[^1] )(以及对 Emacs 感兴趣的人)的,所以我将解释或者说辩解我为什么喜欢它以及我如何使用它。同时我也希望我能介绍清楚,从而让你有个良好的体验,而不是花上几个小时去 Google。
|
||||||
|
|
||||||
|
### 恒久优势
|
||||||
|
|
||||||
|
使用 Emacs 带来的长期优势是让生活更轻松。与最后的收获相比,最开始的付出完全值得。想想这些:
|
||||||
|
|
||||||
|
#### 简单高效
|
||||||
|
|
||||||
|
Org 模式本身就值得花时间,但如果你像我一样,你通常要处理十几份左右的文件 —— 从博客帖子到会议事务清单,再到员工评估。在现代计算世界中,这通常意味着要使用多个应用程序,所有这些程序都有不同的用户界面、保存方式、排序和搜索方式。结果就是你需要不断转换思维环境,记住各种细节。我讨厌在程序间切换,这是在强人所难,因为这是个不完整界面模型[^2] ,并且我讨厌记住本该由计算机记住的东西。在单个环境下,Emacs 对 PHB 甚至比对于程序员更高效,因为程序员更多时候只需要专注于一个程序。转换思维环境的成本比表面上的要更高。操作系统和应用程序厂商已经构建了各种界面,以分散我们对这一现实的注意力。如果你是技术人员,通过快捷键(`M-:`)来访问功能强大的[语言解释器][33]会方便的多[^3] 。
|
||||||
|
|
||||||
|
许多应用程序可以全天全屏地用于编辑文本。但Emacs 是唯一的,因为它既是编辑器也是 Emacs Lisp 解释器。从本质上来说,你工作时只要用电脑上的一两个键就能完成。如果你略懂编程的话,就会发现这代表着你可以在 Emacs 中做 _任何事情_。一旦你在内存中存储了这些指令,你的电脑就可以在工作时几乎实时地为你提供高效的运转。你不会想用 Emacs Lisp 来重建 Excel,因为只要用简单的一两行代码就能实现 Excel 中大多数的功能。比如说我要处理数字,我更有可能转到 scratch 缓冲区,编写一些代码,而不是打开电子表格。即便是要写一封比较大的邮件时,我通常也会先在 Emacs 中写完,然后再复制粘贴到邮件客户端中。当你可以流畅的书写时,为什么要去切换呢?你可以先从一两个简单的算术开始,随着时间的推移,你可以很容易的在 Emacs 中添加你所需要处理的计算。这在应用程序中可能是独一无二的,同时还提供了让为其他的人创造的丰富特性。还记得艾萨克·阿西莫夫书中那些神奇的终端吗? Emacs 是我所遇到的最接近它们的东西[^4] 。我决定不再用什么应用程序来做这个或那个。相反,我只是工作。拥有一个伟大的工具并致力于此,这才是真正的动力和效率。
|
||||||
|
|
||||||
|
#### 静中造物
|
||||||
|
|
||||||
|
拥有所发现的最好的文本编辑功能的最终结果是什么?有一群人在做各种各样有用的补充吗?发挥了 Lisp 键盘的全部威力了吗?我用 Emacs 来完成所有的创作性工作,音乐和图片除外。
|
||||||
|
|
||||||
|
我办公桌上有两个显示器。其中一块竖屏是将 Emacs 全天全屏显示,另一个显示浏览器,用来搜索和阅读,我通常也会打开一个终端。我将日历、邮件等放在 OS X 的另一个桌面上,当我使用 Emacs 时,这个桌面会隐藏起来,同时我也会关掉所有通知。这样就能让我专注于我手头上在做的事了。我发现,越是先进的 UI 应用程序,消除干扰越是不可能,因为这些应用程序致力于提供帮助和易用性。我不需要经常被提醒该如何操作,我已经做了成千上万次了,我真正需要的是一张干净整洁的白纸用来思考。也许因为年龄和自己的“恶习”,我不太喜欢处在嘈杂的环境中,但我认为这值得一试。看看在你电脑环境中有一些真正的宁静是怎样的。当然,现在很多应用程序都有隐藏界面的模式,谢天谢地,苹果和微软现在都有了真正意义上的全屏模式。但是,没有并没有应用程序可以强大到足以“处理”大多数事务。除非你整天写代码,或者像出书一样,处理很长的文档,否则你仍然会面临其他应用程序的干扰。而且,大多数现代应用程序似乎同时显得自视甚高,缺乏功能和可用性[^5] 。比起 office 桌面版,我更讨厌它的在线版。
|
||||||
|
|
||||||
|
![](https://www.fugue.co/hubfs/Imported_Blog_Media/desktop-1.jpg)
|
||||||
|
|
||||||
|
*我的桌面布局, Emacs 在左边*
|
||||||
|
|
||||||
|
但是沟通呢?创造和沟通之间的差别很大。当我将这两件事在不同时间段处理时,我的效率会更高。我们 Fugue 公司使用 Slack,痛并快乐着。我把 Slack 和我的日历、电子邮件放在一个即时通讯的桌面上,这样,当我正在做事时,我就能够忽略所有的聊天信息了。虽然只要一个 Slackstorm 或一封风投或董事会董事的电子邮件,就能让我立刻丢掉手头工作。但是,大多数事情通常可以等上一两个小时。
|
||||||
|
|
||||||
|
#### 普适恒久
|
||||||
|
|
||||||
|
第三个原因是,我发现 Emacs 比其它的环境更有优势的是,你可以很容易地用它来处理事务。我的意思是,你所需要的只是通过类似于 Dropbox 的网站同步一两个目录,而不是让大量的应用程序以它们自己的方式进行交互和同步。然后,你可以在任何你已经精心打造了适合你的目的的套件的环境中工作了。我在 OS X、Windows,或有时在 Linux 都是这样做的。它非常简单可靠。这种功能很有用,以至于我害怕处理 Pages、Google Docs、Office 或其他类型的文件和应用程序,这些文件和应用程序会迫使我回到文件系统或云中的某个地方去寻找。
|
||||||
|
|
||||||
|
限制在计算机上永久存储的因素是文件格式。假设人类已经解决了存储问题[^6] ,随着时间的推移,我们面临的问题是我们能否够继续访问我们创建的信息。文本文件是保存时间最久的格式。你可以用 Emacs 轻松地打开 1970 年的文本文件。然而对于 Office 应用程序却并非如此。同时文本文件要比 Office 应用程序数据文件小得多,也要好的多。作为一个数码背包迷,作为一个在脑子里一闪而过就会做很多小笔记的人,拥有一个简单、轻便、永久、随时可用的东西对我来说很重要。
|
||||||
|
|
||||||
|
如果你准备尝试 Emacs,请继续读下去!下面的部分不是完整的教程,但是在读完后,就可以动手操作了。
|
||||||
|
|
||||||
|
### 驾驭之道 —— 专业定制
|
||||||
|
|
||||||
|
所有这些强大、精神上的平静和安宁的代价是,Emacs 有一个陡峭的学习曲线,它的一切都与你以前所习惯的不同。一开始,这会让你觉得你是在浪费时间在一个过时和奇怪的应用程序上,就好像穿越到过去。这有点像你只开过车,却要你去学骑自行车[^7] 。
|
||||||
|
|
||||||
|
#### 类型抉择
|
||||||
|
|
||||||
|
我用的是来自 GNU 的 OS X 和 Windows 的通用版本的 Emacs。你可以在 [http://emacsformacos.com/][35] 获取 OS X 版本,在 [http://www.gnu.org/software/emacs/][37] 获取 Windows 版本。市面上还有很多其他版本,尤其是 Mac 版本,但我发现,要做一些功能强大的东西(涉及到 Lisp 和许多模式),学习曲线要比实际操作低得多。下载,然后我们就可以开始了[^8] !
|
||||||
|
|
||||||
|
#### 驾驭之始
|
||||||
|
|
||||||
|
在本文中,我将使用 Emacs 的按键和组合键约定。`C` 表示 `Control` 键,`M` 表示 `meta`(通常是 `Alt` 或 `Option` 键),以及用于组合键的连字符。因此,`C-h t` 表示同时按下 `Control` 和 `h` 键,然后释放,再按下 `t`。这个组合快捷键会指向一个教程,这是你首先要做的一件事。
|
||||||
|
|
||||||
|
不要使用方向键或鼠标。它们可以工作,但是你应该给自己一周的时间来使用 Emacs 教程中的原生的导航命令。一旦你这些命令变为了肌肉记忆,你可能就会乐在其中,无论到哪里,你都会非常想念它们。这个 Emacs 教程在介绍它们方面做得很好,但是我将进行总结,所以你不需要阅读全部内容。最无聊的是,不用方向键,用 `C-b` 向前移动,用 `C-f` 向后移动,上一行用 `C-p`,下一行用 `C-n`。你可能会想:“我用方向键就很好,为什么还要这样做?” 有几个原因。首先,你不需要从主键盘区将你的手移开。第二,使用 `Alt`(或用 Emacs 的说法 `Meta`)键来向前或向后在单词间移动。显而易见这样更方便。第三,如果想重复某个命令,可以在命令前面加上一个数字。在编辑文档时,我经常使用这种方法,通过估计向后移动多少个单词或向上或向下移动多少行,然后按下 `C-9 C-p` 或 `M-5 M-b` 之类的快捷键。其它真正重要的导航命令基于开头用 `a` 和结尾用 `e`。在行中使用 `C-a|e`,在句中使用 `M-a|e`。为了让句中的命令正常工作,需要在句号后增加两个空格,这同时提供了一个有用的特性,并消除了脑中一个过时的[观点][38]。如果需要将文档导出到单个空间[发布环境][39],可以编写一个宏来执行此操作。
|
||||||
|
|
||||||
|
Emacs 所附带的教程很值得去看。对于真正缺乏耐心的人,我将介绍一些重要的命令,但那个教程非常有用。记住:用 `C-h t` 进入教程。
|
||||||
|
|
||||||
|
#### 驾驭之复制粘贴
|
||||||
|
|
||||||
|
你可以把 Emacs 设为 CUA 模式,这将会以熟悉的方式工作来操作复制粘贴,但是原生的 Emacs 方法更好,而且你一旦学会了它,就很容易。你可以使用 `Shift` 和导航命令来标记区域(如同选择)。所以 `C-F` 是选中光标前的一个字符,等等。亦可以用 `M-w` 来复制,用 `C-w` 剪切,然后用 `C-y` 粘贴。这些实际上叫做<ruby>删除<rt>killing</rt></ruby>和<ruby>召回<rt>yanking</rt></ruby>,但它非常类似于剪切和粘贴。在删除中还有一些小技巧,但是现在,你只需要关注剪切、复制和粘贴。如果你开始尝试了,那么 `C-x u` 是撤销。
|
||||||
|
|
||||||
|
#### 驾驭之 Ido 模式
|
||||||
|
|
||||||
|
相信我,Ido 会让文件的工作变得很简单。通常,你在 Emacs 中处理文件不需要使用一个单独的访达或文件资源管理器的窗口。相反,你可以用编辑器的命令来创建、打开和保存文件。如果没有 Ido 的话,这将有点麻烦,所以我建议你在学习其他之前安装好它。 Ido 是 Emacs 的 22 版时开始出现的,但是需要对你的 `.emacs` 文件做一些调整,来确保它一直开启着。这是个配置环境的好理由。
|
||||||
|
|
||||||
|
Emacs 中的大多数功能都表现在模式上。要安装指定的模式,需要做两件事。嗯,一开始你需要做一些额外的事情,但这些只需要做一次,然后再做这两件事。那么,这件额外的事情是你需要一个单独的位置来放置所有 Emacs Lisp 文件,并且你需要告诉 Emacs 这个位置在哪。我建议你在 Dropbox 上创建一个单独的目录,那是你 Emacs 主目录。在这里,你需要创建一个 `.emacs` 文件和 `.emacs.d` 目录。在 `.emacs.d` 目录下,创建一个 `lisp` 的目录。就像这样:
|
||||||
|
|
||||||
|
```
|
||||||
|
home
|
||||||
|
|
|
||||||
|
+.emacs
|
||||||
|
|
|
||||||
|
-.emacs.d
|
||||||
|
|
|
||||||
|
-lisp
|
||||||
|
```
|
||||||
|
|
||||||
|
你可以将 `.el` 文件,比如说模式文件,放到 `home/.emacs.d/lisp` 目录下,然后在你的 `.emacs` 文件中添加以下代码来指明该路径:
|
||||||
|
|
||||||
|
```
|
||||||
|
(add-to-list 'load-path "~/.emacs.d/lisp/")
|
||||||
|
```
|
||||||
|
|
||||||
|
Ido 模式是 Emacs 自带的,所以你不需要在你的 `lisp` 目录中放这个 `.el` 文件,但你仍然需要添加上面代码,因为下面的介绍会使用到它.
|
||||||
|
|
||||||
|
#### 驾驭之符号链接
|
||||||
|
|
||||||
|
等等,这里写的 `.emacs` 和 `.emacs.d` 都是存放在你的主目录下,但我们把它们放到了 Dropbox 的某些愚蠢的文件夹!对,这就让你的环境在任何地方都很容易使用。把所有东西都保存在 Dropbox 上,并做符号链接到 `~` 下的 `.emacs` 、`.emacs.d` 和你的主要存放文档的目录。在 OS X 上,使用 `ln -s` 命令非常简单,但在 Windows 上却很麻烦。幸运的是,Emacs 提供了一种简单的方法来替代 Windows 上的符号链接,Windows 的 `HOME` 环境变量。转到 Windows 的环境变量(Windows 10,你可以按 Windows 键然后输入 “环境变量” 来搜索,这是 Windows 10 最好的地方了),在你的帐户下创建一个指向你在 Dropbox 中 Emacs 的文件夹的 `HOME` 环境变量。如果你想方便地浏览 Dropbox 之外的本地文件,你可能想在你的实际主目录下建立一个到 Dropbox 下 Emacs 主目录的符号链接。
|
||||||
|
|
||||||
|
至此,你已经完成了在任意机器上指向你的 Emacs 配置和文件所需的技巧。如果你买了一台新电脑,或者用别人的电脑一小时或一天,你就可以得到你的整个工作环境。第一次操作起来似乎有点难,但是一旦你知道你在做什么,就(最多)只需要 10 分钟。
|
||||||
|
|
||||||
|
但我们现在是在配置 Ido ……
|
||||||
|
|
||||||
|
按下 `C-x` `C-f` 然后输入 `~/.emacs` 和两次回车来创建 `.emacs` 文件,将下面几行添加进去:
|
||||||
|
|
||||||
|
```
|
||||||
|
;; set up ido mode
|
||||||
|
(require `ido)
|
||||||
|
(setq ido-enable-flex-matching t)
|
||||||
|
(setq ido-everywhere t)
|
||||||
|
(ido-mode 1)
|
||||||
|
```
|
||||||
|
|
||||||
|
在 `.emacs` 窗口开着的时候,执行 `M-x evaluate-buffer` 命令。如果某处弄错了的话,将得到一个错误,或者你将得到 Ido。Ido 改变了在 minibuffer 中操作文件操方式。关于这个有一篇比较好的文档,但是我也会指出一些技巧。有效地使用 `~/`;你可以在 minibuffer 的任何地方输入 `~/`,它就会跳转到主目录。这就意味着,你应该让你的大部分东西就近的放在主目录下。我用 `~/org` 目录来保存所有非代码的东西,用 `~/code` 保存代码。一旦你进入到正确的目录,通常会拥有一组具有不同扩展名的文件,特别是当你使用 Org 模式并从中发布的话。你可以输入 `.` 和想要的扩展名,无论你的在文件名的什么位置,Ido 都会将选择限制在具有该扩展名的文件中。例如,我在 Org 模式下写这篇博客,所以该文件是:
|
||||||
|
|
||||||
|
```
|
||||||
|
~/org/blog/emacs.org
|
||||||
|
```
|
||||||
|
|
||||||
|
我偶尔也会用 Org 模式发布成 HTML 格式,所以我将在同一目录下得到 `emacs.html` 文件。当我想打开该 Org 文件时,我会输入:
|
||||||
|
|
||||||
|
```
|
||||||
|
C-x C-f ~/o[RET]/bl[RET].or[RET]
|
||||||
|
```
|
||||||
|
|
||||||
|
其中 `[RET]` 是我使用 `Ido` 模式的自动补全而按下的回车键。所以,这只需要按 12 个键,如果你习惯了的话, 这将比打开访达或文件资源管理器再用鼠标点要节省 _很_ 多时间。 Ido 模式很有用,而这只是操作 Emacs 的一种实用模式而已。下面让我们去探索一些其它对完成工作很有帮助的模式吧。
|
||||||
|
|
||||||
|
#### 驾驭之字体风格
|
||||||
|
|
||||||
|
我推荐在 Emacs 中使用漂亮的字体族。它们可以使用不同的括号、0 和其他字符进行自定义。你可以在字体文件本身中构建额外的行间距。我推荐 1.5 倍的行间距,并在代码和数据中使用不等宽字体。写作中我用 `Serif` 字体,它有一种紧凑但时髦的感觉。你可以在 [http://input.fontbureau.com/][40] 上找到它们,在那里你可以根据自己的喜好进行定制。你可以使用 Emacs 中的菜单手动设置字体,但这会将代码保存到你的 `.emacs` 文件中,如果你使用多个设备,你可能需要一些不同的设置。我将我的 `.emacs` 设置为根据使用的机器的名称来相应配置屏幕。代码如下:
|
||||||
|
|
||||||
|
```
|
||||||
|
;; set up fonts for different OSes. OSX toggles to full screen.
|
||||||
|
(setq myfont "InputSerif")
|
||||||
|
(cond
|
||||||
|
((string-equal system-name "Sampo.local")
|
||||||
|
(set-face-attribute 'default nil :font myfont :height 144)
|
||||||
|
(toggle-frame-fullscreen))
|
||||||
|
((string-equal system-name "Morpheus.local")
|
||||||
|
(set-face-attribute 'default nil :font myfont :height 144))
|
||||||
|
((string-equal system-name "ILMARINEN")
|
||||||
|
(set-face-attribute 'default nil :font myfont :height 106))
|
||||||
|
((string-equal system-name "UKKO")
|
||||||
|
(set-face-attribute 'default nil :font myfont :height 104)))
|
||||||
|
```
|
||||||
|
|
||||||
|
你应该将 Emacs 中的 `system-name` 的值替换成你通过 `(system-name)` 得到的值。注意,在 Sampo (我的 MacBook)上,我还将 Emacs 设置为全屏。我也想在 Windows 实现这个功能,但是 Windows 和 Emacs 好像互相嫌弃对方,当我尝试配置时,它总是不稳定。相反,我只能在启动后手动全屏。
|
||||||
|
|
||||||
|
我还建议去掉 Emacs 中的上世纪 90 年代出现的难看工具栏,当时比较流行在应用程序中使用工具栏。我还去掉了一些其它的“电镀层”,这样我就有了一个简单、高效的界面。把这些加到你的 `.emacs` 的文件中来去掉工具栏和滚动条,但要保留菜单(在 OS X 上,它将被隐藏,除非你将鼠标到屏幕顶部):
|
||||||
|
|
||||||
|
```
|
||||||
|
(if (fboundp 'scroll-bar-mode) (scroll-bar-mode -1))
|
||||||
|
(if (fboundp 'tool-bar-mode) (tool-bar-mode -1))
|
||||||
|
(if (fboundp 'menu-bar-mode) (menu-bar-mode 1))
|
||||||
|
```
|
||||||
|
|
||||||
|
#### 驾驭之 Org 模式
|
||||||
|
|
||||||
|
我基本上是在 Org 模式下处理工作的。它是我创作文档、记笔记、列任务清单以及 90% 其他工作的首选环境。Org 模式是笔记和待办事项列表的组合工具,最初是由一个在会议中使用笔记本电脑的人构想出来的。我反对在会议中使用笔记本电脑,自己也不使用,所以我的用法与他的有些不同。对我来说,Org 模式主要是一种处理结构中内容的方式。在 Org 模式中有标题和副标题等,它们的作用就像一个大纲。Org 模式允许你展开或隐藏大纲树,还可以重新排列该树。这正合我意,并且我发现用这种方式使用它是一种乐趣。
|
||||||
|
|
||||||
|
Org 模式也有很多让生活愉快的小功能。例如,脚注处理非常好,LaTeX/PDF 输出也很好。Org 模式能够根据所有文档中的待办事项生成议程,并能很好地将它们与日期/时间联系起来。我不把它用在任何形式的外部任务上,这些任务都是在一个共享的日历上处理的,但是在创建事物和跟踪我未来需要创建的东西时,它是无价的。安装它,你只要将 `org-mode.el` 放到你的 `lisp` 目录下。如果你想要它基于文档的结构进行缩进并在打开时全部展开的话,在你的 `.emacs` 文件中添加如下代码:
|
||||||
|
|
||||||
|
```
|
||||||
|
;; set up org mode
|
||||||
|
(setq org-startup-indented t)
|
||||||
|
(setq org-startup-folded "showall")
|
||||||
|
(setq org-directory "~/org")
|
||||||
|
```
|
||||||
|
|
||||||
|
最后一行是让 Org 模式知道在哪里查找要包含在议程和其他事情中的文件。我把 Org 模式保存在我的主目录中,也就是说,像前面介绍的一样,它是 Dropbox 目录的一个符号链接。
|
||||||
|
|
||||||
|
我有一个总是在缓冲区中打开的 `stuff.org` 文件。我把它当作记事本。Org 模式使得提取待办事项和有期限的事情变得很容易。当你能够内联 Lisp 代码并在需要计算它时,它特别有用。拥有包含内容的代码非常方便。同样,你可以使用 Emacs 访问实际的计算机,这是一种解放。
|
||||||
|
|
||||||
|
##### 用 Org 模式进行发布
|
||||||
|
|
||||||
|
我关心的是文档的外观及格式。我刚开始工作时是个设计师,而且我认为信息可以,也应该表现得清晰和美丽。Org 模式对将 LaTeX 生成 PDF 支持的很好,LaTeX 虽然也有学习曲线,但是很容易处理一些简单的事务。
|
||||||
|
|
||||||
|
如果你想使用字体和样式,而不是典型的 LaTeX 字体和样式,你需要做些事。首先,你要用到 XeLaTeX,这样就可以使用普通的系统字体,而不是 LaTeX 的特殊字体。接下来,你需要将以下代码添加到 `.emacs` 中:
|
||||||
|
|
||||||
|
```
|
||||||
|
(setq org-latex-pdf-process
|
||||||
|
'("xelatex -interaction nonstopmode %f"
|
||||||
|
"xelatex -interaction nonstopmode %f"))
|
||||||
|
```
|
||||||
|
|
||||||
|
我把这个放在 `.emacs` 中 Org 模式配置部分的末尾,使文档变得更整洁。这让你在从 Org 模式发布时可以使用更多格式化选项。例如,我经常使用:
|
||||||
|
|
||||||
|
```
|
||||||
|
#+LaTeX_HEADER: \usepackage{fontspec}
|
||||||
|
#+LATEX_HEADER: \setmonofont[Scale=0.9]{Input Mono}
|
||||||
|
#+LATEX_HEADER: \setromanfont{Maison Neue}
|
||||||
|
#+LATEX_HEADER: \linespread{1.5}
|
||||||
|
#+LATEX_HEADER: \usepackage[margin=1.25in]{geometry}
|
||||||
|
|
||||||
|
#+TITLE: Document Title Here
|
||||||
|
```
|
||||||
|
|
||||||
|
这些都可以在 `.org` 文件中找到。我们的公司规定的正文字体是 `Maison Neue`,但你也可以在这写上任何适当的东西。我很是抵制 `Maison Neue`,因为这是一种糟糕的字体,任何人都不应该使用它。
|
||||||
|
|
||||||
|
这个文件是一个使用该配置输出为 PDF 的实例。这就是开箱即用的 LaTeX 一样。在我看来这还不错,但是字体很平淡,而且有点奇怪。此外,如果你使用标准格式,人们会觉得他们正在阅读的东西是、或者假装是一篇学术论文。别怪我没提醒你。
|
||||||
|
|
||||||
|
#### 驾驭之 Ace Jump 模式
|
||||||
|
|
||||||
|
这只是一个辅助模式,而不是一个主模式,但是你也需要它。其工作原理有点像之前提到的 Jef Raskin 的 Leap 功能[^9] 。 按下 `C-c C-SPC`,然后输入要跳转到单词的第一个字母。它会高亮显示所有以该字母开头的单词,并将其替换为字母表中的字母。你只需键入所需位置的字母,光标就会跳转到该位置。我常将它作为导航键或是用来检索。将 `.el` 文件下到你的 `lisp` 目录下,并在 `.emacs` 文件添加如下代码:
|
||||||
|
|
||||||
|
```
|
||||||
|
;; set up ace-jump-mode
|
||||||
|
(add-to-list 'load-path "which-folder-ace-jump-mode-file-in/")
|
||||||
|
(require 'ace-jump-mode)
|
||||||
|
(define-key global-map (kbd "C-c C-SPC" ) 'ace-jump-mode)
|
||||||
|
```
|
||||||
|
|
||||||
|
### 待续
|
||||||
|
|
||||||
|
本文已经够详细了,你能在其中得到你所想要的。我很想知道除编程之外(或用于编程)Emacs 的使用情况,及其是否高效。在我使用 Emacs 的过程中,可能存在一些自作聪明的老板式想法,如果你能指出来,我将不胜感激。之后,我可能会写一些更新来介绍其它特性或模式。我很确定我将会向你展示如何在 Emacs 和 Ludwig 模式下使用 Fugue,因为我会将它发展成比代码高亮更有用的东西。更多想法,请在 Twitter 上 [@fugueHQ][41] 。
|
||||||
|
|
||||||
|
### 脚注
|
||||||
|
|
||||||
|
[^1]: 如果你是位精英,但从没涉及过技术方面,那么 Emacs 并不适合你。对于少数的人来说,Emacs 可能会为他们开辟一条通往计算机技术方面的道路,但这只是极少数。如果你知道怎么使用 Unix 或 Windows 的终端,或者曾编辑过 dotfile,或者说你曾写过一点代码的话,这对使用 Emacs 有很大的帮助。
|
||||||
|
|
||||||
|
[^2]: 参考链接: http://archive.wired.com/wired/archive/2.08/tufte.html
|
||||||
|
|
||||||
|
[^3]: 我主要是在写作时使用这个模式来进行一些运算。比如说,当我在给一个新雇员写一封入职信时,我想要算这封入职信中有多少个选项。由于我在我的 `.emacs` 为 outstanding-shares 定义了一个变量,所以我只要按下 `M-:` 然后输入 `(* .001 outstanding-shares)` 就能再无需打开计算器或电子表格的情况下得到精度为 0.001 的结果。我使用了 _大量_ 的变量来避免程序间切换。
|
||||||
|
|
||||||
|
[^4]: 缺少的部分是 web。有个名为 eww 的 Emacs 网页浏览器能够让你在 Emacs 中浏览网页。我用的就是这个,因为它既能拦截广告(LCTT 译注:实质上是无法显示,/laugh),同时也在可读性方面为 web 开发者消除了大多数差劲的选项。这个其实有点类似于 Safari 的阅读模式。不幸的是,大部分网站都有很多令人讨厌的繁琐的东西以及难以转换为文本的导航,
|
||||||
|
|
||||||
|
[^5]: 易用性和易学性这两者经常容易被搞混。易学性是指学习使用工具的难易程度。而易用性是指工具高效的程度。通常来说,这是要差别的,就想鼠标和菜单栏的差别一样。菜单栏很容易学会,但是却不怎么高效,以致于早期会存在一些键盘的快捷键。除了在 GUI 方面上,Raskin 在很多方面上的观点都很正确。如今,操作系统正在将一些合适的搜索映射到键盘的快捷键上。比如说在 OS X 和 Windows 上,我默认的导航方式就是搜索。Ubuntu 的搜索做的很差劲,如同它的 GUI 一样差劲。
|
||||||
|
|
||||||
|
[^6]: 在有网的情况下,[AWS S3][42] 是解决文件存储问题的有效方案。数万亿个对象存在 S3 中,但是从来没有遗失过。大部分提供云存储的服务都是在 S3 上或是模拟 S3 构建的。没人能够拥有 S3 一样的规模,所以我将重要的文件通过 Dropbox 存储在上面。
|
||||||
|
|
||||||
|
[^7]: 目前,你可能会想:“这个人和自行车有什么关系?”……我在各个层面上都喜欢自行车。自行车是迄今为止发明的最具机械效率的交通工具。自行车可以是真正美丽的事物。而且,只要注意点的话,自行车可以用一辈子。早在 2001 年,我曾向 Rivendell Bicycle Works 订购了一辆自行车,现在我每次看到那辆自行车依然很高兴,自行车和 Unix 是我接触过的最好的两个发明。对了,还有 Emacs。
|
||||||
|
|
||||||
|
[^8]: 这个网站有一个很棒的 Emacs 教程,但不是这个。当我浏览这个页面时,我确实得到了一些对获取高效的 Emacs 配置很重要的知识,但无论怎么说,这都不是个替代品。
|
||||||
|
|
||||||
|
[^9]: 20 世纪 80 年代,Jef Raskin 与 Steve Jobs 在 Macintosh 项目上闹翻后, Jef Raskin 又设计了 [Canon Cat 计算机][43]。这台 Cat 计算机是以文档为中心的界面(所有的计算机都应如此),并以一种全新的方式使用键盘,你现在可以用 Emacs 来模仿这种键盘。如果现在有一台现代的,功能强大的 Cat 计算机并配有一个高分辨的显示器和 Unix 系统的话,我立马会用 Mac 来换。[https://youtu.be/o_TlE_U_X3c?t=19s][28]
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: https://blog.fugue.co/2015-11-11-guide-to-emacs.html
|
||||||
|
|
||||||
|
作者:[Josh Stella][a]
|
||||||
|
译者:[oneforalone](https://github.com/oneforalone)
|
||||||
|
校对:[wxy](https://github.com/wxy), [oneforalone](https://github.com/oneforalone)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]:https://blog.fugue.co/authors/josh.html
|
||||||
|
[1]:https://blog.fugue.co/2013-10-16-vpc-on-aws-part3.html
|
||||||
|
[2]:https://blog.fugue.co/2013-10-02-vpc-on-aws-part2.html
|
||||||
|
[3]:http://ww2.fugue.co/2017-05-25_OS_AR_GartnerCoolVendor2017_01-LP-Registration.html
|
||||||
|
[4]:https://blog.fugue.co/authors/josh.html
|
||||||
|
[5]:https://twitter.com/joshstella
|
||||||
|
[6]:https://www.youtube.com/watch?v=khJQgRLKMU0
|
||||||
|
[7]:https://blog.fugue.co/2015-11-11-guide-to-emacs.html?utm_source=wanqu.co&utm_campaign=Wanqu+Daily&utm_medium=website#phb
|
||||||
|
[8]:https://blog.fugue.co/2015-11-11-guide-to-emacs.html?utm_source=wanqu.co&utm_campaign=Wanqu+Daily&utm_medium=website#tufte
|
||||||
|
[9]:https://blog.fugue.co/2015-11-11-guide-to-emacs.html?utm_source=wanqu.co&utm_campaign=Wanqu+Daily&utm_medium=website#interpreter
|
||||||
|
[10]:https://blog.fugue.co/2015-11-11-guide-to-emacs.html?utm_source=wanqu.co&utm_campaign=Wanqu+Daily&utm_medium=website#eww
|
||||||
|
[11]:https://blog.fugue.co/2015-11-11-guide-to-emacs.html?utm_source=wanqu.co&utm_campaign=Wanqu+Daily&utm_medium=website#usability
|
||||||
|
[12]:https://blog.fugue.co/2015-11-11-guide-to-emacs.html?utm_source=wanqu.co&utm_campaign=Wanqu+Daily&utm_medium=website#s3
|
||||||
|
[13]:https://blog.fugue.co/2015-11-11-guide-to-emacs.html?utm_source=wanqu.co&utm_campaign=Wanqu+Daily&utm_medium=website#bicycles
|
||||||
|
[14]:https://blog.fugue.co/2015-11-11-guide-to-emacs.html?utm_source=wanqu.co&utm_campaign=Wanqu+Daily&utm_medium=website#nottutorial
|
||||||
|
[15]:https://blog.fugue.co/2015-11-11-guide-to-emacs.html?utm_source=wanqu.co&utm_campaign=Wanqu+Daily&utm_medium=website#canoncat
|
||||||
|
[16]:https://blog.fugue.co/2015-11-11-guide-to-emacs.html?utm_source=wanqu.co&utm_campaign=Wanqu+Daily&utm_medium=website#phbOrigin
|
||||||
|
[17]:https://blog.fugue.co/2015-11-11-guide-to-emacs.html?utm_source=wanqu.co&utm_campaign=Wanqu+Daily&utm_medium=website#tufteOrigin
|
||||||
|
[18]:http://archive.wired.com/wired/archive/2.08/tufte.html
|
||||||
|
[19]:http://archive.wired.com/wired/archive/2.08/tufte.html
|
||||||
|
[20]:https://blog.fugue.co/2015-11-11-guide-to-emacs.html?utm_source=wanqu.co&utm_campaign=Wanqu+Daily&utm_medium=website#interpreterOrigin
|
||||||
|
[21]:https://blog.fugue.co/2015-11-11-guide-to-emacs.html?utm_source=wanqu.co&utm_campaign=Wanqu+Daily&utm_medium=website#ewwOrigin
|
||||||
|
[22]:https://blog.fugue.co/2015-11-11-guide-to-emacs.html?utm_source=wanqu.co&utm_campaign=Wanqu+Daily&utm_medium=website#usabilityOrigin
|
||||||
|
[23]:https://blog.fugue.co/2015-11-11-guide-to-emacs.html?utm_source=wanqu.co&utm_campaign=Wanqu+Daily&utm_medium=website#s3Origin
|
||||||
|
[24]:https://blog.fugue.co/2015-11-11-guide-to-emacs.html?utm_source=wanqu.co&utm_campaign=Wanqu+Daily&utm_medium=website#bicyclesOrigin
|
||||||
|
[25]:https://blog.fugue.co/2015-11-11-guide-to-emacs.html?utm_source=wanqu.co&utm_campaign=Wanqu+Daily&utm_medium=website#nottutorialOrigin
|
||||||
|
[26]:https://blog.fugue.co/2015-11-11-guide-to-emacs.html?utm_source=wanqu.co&utm_campaign=Wanqu+Daily&utm_medium=website#canoncatOrigin
|
||||||
|
[27]:https://youtu.be/o_TlE_U_X3c?t=19s
|
||||||
|
[28]:https://youtu.be/o_TlE_U_X3c?t=19s
|
||||||
|
[29]:https://blog.fugue.co/authors/josh.html
|
||||||
|
[30]:http://www.huffingtonpost.com/zachary-ehren/soma-isnt-a-drug-san-fran_b_987841.html
|
||||||
|
[31]:http://www.campagnolo.com/US/en
|
||||||
|
[32]:http://www.businessinsider.com/best-pointy-haired-boss-moments-from-dilbert-2013-10
|
||||||
|
[33]:http://www.webopedia.com/TERM/I/interpreter.html
|
||||||
|
[34]:http://emacsformacosx.com/
|
||||||
|
[35]:http://emacsformacosx.com/
|
||||||
|
[36]:http://www.gnu.org/software/emacs/
|
||||||
|
[37]:http://www.gnu.org/software/emacs/
|
||||||
|
[38]:http://www.huffingtonpost.com/2015/05/29/two-spaces-after-period-debate_n_7455660.html
|
||||||
|
[39]:http://practicaltypography.com/one-space-between-sentences.html
|
||||||
|
[40]:http://input.fontbureau.com/
|
||||||
|
[41]:https://twitter.com/fugueHQ
|
||||||
|
[42]:https://baike.baidu.com/item/amazon%20s3/10809744?fr=aladdin
|
||||||
|
[43]:https://en.wikipedia.org/wiki/Canon_Cat
|
@ -1,24 +1,38 @@
|
|||||||
[#]: collector: (lujun9972)
|
[#]: collector: (lujun9972)
|
||||||
[#]: translator: (jlztan)
|
[#]: translator: (jlztan)
|
||||||
[#]: reviewer: ( )
|
[#]: reviewer: (wxy)
|
||||||
[#]: publisher: ( )
|
[#]: publisher: (wxy)
|
||||||
[#]: subject: (Celebrate Christmas In Linux Way With These Wallpapers)
|
[#]: subject: (Celebrate Christmas In Linux Way With These Wallpapers)
|
||||||
[#]: via: (https://itsfoss.com/christmas-linux-wallpaper/)
|
[#]: via: (https://itsfoss.com/christmas-linux-wallpaper/)
|
||||||
[#]: author: (Abhishek Prakash https://itsfoss.com/author/abhishek/)
|
[#]: author: (Abhishek Prakash https://itsfoss.com/author/abhishek/)
|
||||||
[#]: url: ( )
|
[#]: url: (https://linux.cn/article-10381-1.html)
|
||||||
|
|
||||||
使用这些壁纸以 Linux 的方式庆祝圣诞节
|
以 Linux 的方式庆祝圣诞节
|
||||||
======
|
======
|
||||||
|
|
||||||
当前正是假日季节,很多人可能已经在庆祝圣诞节了。我代表 It's FOSS 团队祝你圣诞快乐,新年快乐。
|
当前正是假日季,很多人可能已经在庆祝圣诞节了。祝你圣诞快乐,新年快乐。
|
||||||
|
|
||||||
为了延续节日氛围,我将向你展示一些非常棒的圣诞主题 [Linux壁纸][1]。在呈现这些壁纸之前,先来看一棵 Linux 终端下的圣诞树。
|
为了延续节日氛围,我将向你展示一些非常棒的圣诞主题的 [Linux 壁纸][1]。在呈现这些壁纸之前,先来看一棵 Linux 终端下的圣诞树。
|
||||||
|
|
||||||
|
### 让你的桌面飘雪(针对 GNOME 用户)
|
||||||
|
|
||||||
|
- [Let it Snow on Your Linux Desktop](https://youtu.be/1QI1ludzZuA)
|
||||||
|
|
||||||
|
如果您在 Ubuntu 18.04 或任何其他 Linux 发行版中使用 GNOME 桌面,您可以使用一个小的 [GNOME 扩展][55]并在桌面上飘雪。
|
||||||
|
|
||||||
|
您可以从软件中心或 GNOME 扩展网站获取此 gsnow 扩展。我建议您阅读一些关于[使用 GNOME 扩展][55]的内容。
|
||||||
|
|
||||||
|
安装此扩展程序后,您会在顶部面板上看到一个小雪花图标。 如果您单击一次,您会看到桌面屏幕上的小絮状物掉落。
|
||||||
|
|
||||||
|
![](https://itsfoss.com/wp-content/uploads/2018/12/snowfall-on-linux-desktop-1.webm)
|
||||||
|
|
||||||
|
你可以再次点击该图标来禁止雪花落下。
|
||||||
|
|
||||||
### 在 Linux 终端下显示圣诞树
|
### 在 Linux 终端下显示圣诞树
|
||||||
|
|
||||||
圣诞树的效果如 [这个页面](https://giphy.com/embed/xUNda6KphvbpYxL3tm) 所示。
|
![Display Christmas Tree in Linux Terminal](https://i.giphy.com/xUNda6KphvbpYxL3tm.gif)
|
||||||
|
|
||||||
使用以下命令在终端中显示一棵动画圣诞树:
|
如果你想要在终端里显示一个动画的圣诞树,你可以使用如下命令:
|
||||||
|
|
||||||
```
|
```
|
||||||
curl https://raw.githubusercontent.com/sergiolepore/ChristBASHTree/master/tree-EN.sh | bash
|
curl https://raw.githubusercontent.com/sergiolepore/ChristBASHTree/master/tree-EN.sh | bash
|
||||||
@ -40,7 +54,7 @@ perl -MCPAN -e 'install Acme::POE::Tree'
|
|||||||
|
|
||||||
你可以阅读 [原文][5] 来了解更多信息。
|
你可以阅读 [原文][5] 来了解更多信息。
|
||||||
|
|
||||||
## 下载 Linux 圣诞主题壁纸
|
### 下载 Linux 圣诞主题壁纸
|
||||||
|
|
||||||
所有这些 Linux 圣诞主题壁纸都是由 Mark Riedesel 制作的,你可以在 [他的网站][6] 上找到很多其他艺术品。
|
所有这些 Linux 圣诞主题壁纸都是由 Mark Riedesel 制作的,你可以在 [他的网站][6] 上找到很多其他艺术品。
|
||||||
|
|
||||||
@ -48,87 +62,95 @@ perl -MCPAN -e 'install Acme::POE::Tree'
|
|||||||
|
|
||||||
注意一个小地方,这里显示的图片都是高度压缩的,因此你要通过图片下方提供的链接进行下载。
|
注意一个小地方,这里显示的图片都是高度压缩的,因此你要通过图片下方提供的链接进行下载。
|
||||||
|
|
||||||
|
![Christmas Linux Wallpaper][56]
|
||||||
|
|
||||||
|
*[下载此壁纸][57]*
|
||||||
|
|
||||||
![Christmas Linux Wallpaper][7]
|
![Christmas Linux Wallpaper][7]
|
||||||
|
|
||||||
[下载此壁纸][8]
|
*[下载此壁纸][8]*
|
||||||
|
|
||||||
[![Christmas Linux Wallpapers][9]][10]
|
[![Christmas Linux Wallpapers][9]][10]
|
||||||
|
|
||||||
[下载此壁纸][11]
|
*[下载此壁纸][11]*
|
||||||
|
|
||||||
[![Christmas Linux Wallpapers][12]][13]
|
[![Christmas Linux Wallpapers][12]][13]
|
||||||
|
|
||||||
[下载此壁纸][14]
|
*[下载此壁纸][14]*
|
||||||
|
|
||||||
[![Christmas Linux Wallpapers][15]][16]
|
[![Christmas Linux Wallpapers][15]][16]
|
||||||
|
|
||||||
[下载此壁纸][17]
|
*[下载此壁纸][17]*
|
||||||
|
|
||||||
[![Christmas Linux Wallpapers][18]][19]
|
[![Christmas Linux Wallpapers][18]][19]
|
||||||
|
|
||||||
[下载此壁纸][20]
|
*[下载此壁纸][20]*
|
||||||
|
|
||||||
[![Christmas Linux Wallpapers][21]][22]
|
[![Christmas Linux Wallpapers][21]][22]
|
||||||
|
|
||||||
[下载此壁纸][23]
|
*[下载此壁纸][23]*
|
||||||
|
|
||||||
[![Christmas Linux Wallpapers][24]][25]
|
[![Christmas Linux Wallpapers][24]][25]
|
||||||
|
|
||||||
[下载此壁纸][26]
|
*[下载此壁纸][26]*
|
||||||
|
|
||||||
[![Christmas Linux Wallpapers][27]][28]
|
[![Christmas Linux Wallpapers][27]][28]
|
||||||
|
|
||||||
[下载此壁纸][29]
|
*[下载此壁纸][29]*
|
||||||
|
|
||||||
[![Christmas Linux Wallpapers][30]][31]
|
[![Christmas Linux Wallpapers][30]][31]
|
||||||
|
|
||||||
[下载此壁纸][32]
|
*[下载此壁纸][32]*
|
||||||
|
|
||||||
[![Christmas Linux Wallpapers][33]][34]
|
[![Christmas Linux Wallpapers][33]][34]
|
||||||
|
|
||||||
[下载此壁纸][35]
|
*[下载此壁纸][35]*
|
||||||
|
|
||||||
[![Christmas Linux Wallpapers][36]][37]
|
[![Christmas Linux Wallpapers][36]][37]
|
||||||
|
|
||||||
[下载此壁纸][38]
|
*[下载此壁纸][38]*
|
||||||
|
|
||||||
[![Christmas Linux Wallpapers][39]][40]
|
[![Christmas Linux Wallpapers][39]][40]
|
||||||
|
|
||||||
[下载此壁纸][41]
|
*[下载此壁纸][41]*
|
||||||
|
|
||||||
[![Christmas Linux Wallpapers][42]][43]
|
[![Christmas Linux Wallpapers][42]][43]
|
||||||
|
|
||||||
[下载此壁纸][44]
|
*[下载此壁纸][44]*
|
||||||
|
|
||||||
[![Christmas Linux Wallpapers][45]][46]
|
[![Christmas Linux Wallpapers][45]][46]
|
||||||
|
|
||||||
[下载此壁纸][47]
|
*[下载此壁纸][47]*
|
||||||
|
|
||||||
[![Christmas Linux Wallpapers][48]][49]
|
[![Christmas Linux Wallpapers][48]][49]
|
||||||
|
|
||||||
[下载此壁纸][50]
|
*[下载此壁纸][50]*
|
||||||
|
|
||||||
### 福利:Linux 圣诞颂歌
|
### 福利:Linux 圣诞颂歌
|
||||||
|
|
||||||
这是给你的一份福利,给像我们一样的 Linux 爱好者的关于 Linux 的圣诞颂歌。
|
这是给你的一份福利,给像我们一样的 Linux 爱好者的关于 Linux 的圣诞颂歌。
|
||||||
|
|
||||||
在 [计算机世界的一篇文章][51] 中,[Sandra Henry-Stocker][52] 分享了这些圣诞颂歌。摘录片段如下:
|
在 [《计算机世界》的一篇文章][51] 中,[Sandra Henry-Stocker][52] 分享了这些圣诞颂歌。摘录片段如下:
|
||||||
|
|
||||||
这一段用的 [Chestnuts Roasting on an Open Fire][53] 的曲调:
|
这一段用的 [Chestnuts Roasting on an Open Fire][53] 的曲调:
|
||||||
|
|
||||||
> Running merrily on open source
|
> Running merrily on open source
|
||||||
|
>
|
||||||
> With users happy as can be
|
> With users happy as can be
|
||||||
|
>
|
||||||
> We’re using Linux and getting lots done
|
> We’re using Linux and getting lots done
|
||||||
|
|
||||||
> And happy everything is free
|
> And happy everything is free
|
||||||
|
|
||||||
这一段用的 [The Twelve Days of Christmas][54] 的曲调:
|
这一段用的 [The Twelve Days of Christmas][54] 的曲调:
|
||||||
|
|
||||||
> On my first day with Linux, my admin gave to me a password and a login ID
|
> On my first day with Linux, my admin gave to me a password and a login ID
|
||||||
|
>
|
||||||
> On my second day with Linux my admin gave to me two new commands and a password and a login ID
|
> On my second day with Linux my admin gave to me two new commands and a password and a login ID
|
||||||
|
|
||||||
在 [这里][51] 阅读完整的颂歌。
|
在 [这里][51] 阅读完整的颂歌。
|
||||||
|
|
||||||
祝你享受 Linux!
|
Linux 快乐!
|
||||||
|
|
||||||
------
|
------
|
||||||
|
|
||||||
@ -137,7 +159,7 @@ via: https://itsfoss.com/christmas-linux-wallpaper/
|
|||||||
作者:[Abhishek Prakash][a]
|
作者:[Abhishek Prakash][a]
|
||||||
选题:[lujun9972][b]
|
选题:[lujun9972][b]
|
||||||
译者:[jlztan](https://github.com/jlztan)
|
译者:[jlztan](https://github.com/jlztan)
|
||||||
校对:[校对者ID](https://github.com/校对者ID)
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
@ -197,3 +219,6 @@ via: https://itsfoss.com/christmas-linux-wallpaper/
|
|||||||
[52]: https://twitter.com/bugfarm
|
[52]: https://twitter.com/bugfarm
|
||||||
[53]: https://www.youtube.com/watch?v=dhzxQCTCI3E
|
[53]: https://www.youtube.com/watch?v=dhzxQCTCI3E
|
||||||
[54]: https://www.youtube.com/watch?v=oyEyMjdD2uk
|
[54]: https://www.youtube.com/watch?v=oyEyMjdD2uk
|
||||||
|
[55]: https://itsfoss.com/gnome-shell-extensions/
|
||||||
|
[56]: https://i0.wp.com/itsfoss.com/wp-content/uploads/2016/12/ChristmasTux2018.jpeg?w=800&ssl=1
|
||||||
|
[57]: http://www.klowner.com/wallery/christmas_tux_2018/download/ChristmasTux2018_4K_3840x2160.png
|
@ -0,0 +1,240 @@
|
|||||||
|
我的个人电子邮件系统设置:notmuch、mbsync、Postfix 和 dovecot
|
||||||
|
======
|
||||||
|
|
||||||
|
我使用个人电子邮件系统已经相当长的时间了,但是一直没有记录过文档。最近我换了我的笔记本电脑(职业变更导致的变动),我在试图重新创建本地邮件系统时迷茫了。所以这篇文章是一个给自己看的文档,这样我就不用费劲就能再次搭建出来。
|
||||||
|
|
||||||
|
### 服务器端
|
||||||
|
|
||||||
|
我运行自己的邮件服务器,并使用 Postfix 作为 SMTP 服务器,用 Dovecot 实现 IMAP。我不打算详细介绍如何配置这些设置,因为我的设置主要是通过使用 Jonas 为 Redpill 基础架构创建的脚本完成的。什么是 Redpill?(用 Jonas 自己的话说):
|
||||||
|
|
||||||
|
> \<jonas> Redpill 是一个概念:一种设置 Debian hosts 去跨组织协作的方式
|
||||||
|
>
|
||||||
|
> \<jonas> 我发展了这个概念,并将其首次用于 Redpill 网中网:redpill.dk,其中涉及到了我自己的网络(jones.dk),我的主要客户的网络(homebase.dk),一个包括 Skolelinux Germany(free-owl.de)的在德国的网络,和 Vasudev 的网络(copyninja.info)
|
||||||
|
|
||||||
|
除此之外, 我还有一个 dovecot sieve 过滤,根据邮件的来源,对邮件进行高级分类,将其放到各种文件夹中。所有的规则都存在于每个有邮件地址的账户下的 `~/dovecot.sieve` 文件中。
|
||||||
|
|
||||||
|
再次,我不会详细介绍如何设置这些东西,因为这不是我这个帖子的目标。
|
||||||
|
|
||||||
|
### 在我的笔记本电脑上
|
||||||
|
|
||||||
|
在我的笔记本电脑上,我已经按照 4 个部分设置
|
||||||
|
|
||||||
|
1. 邮件同步:使用 `mbsync` 命令完成
|
||||||
|
2. 分类:使用 notmuch 完成
|
||||||
|
3. 阅读:使用 notmuch-emacs 完成
|
||||||
|
4. 邮件发送:使用作为中继服务器和 SMTP 客户端运行的 Postfix 完成。
|
||||||
|
|
||||||
|
### 邮件同步
|
||||||
|
|
||||||
|
邮件同步是使用 `mbsync` 工具完成的, 我以前是 OfflineIMAP 的用户,最近切换到 `mbsync`,因为我觉得它比 OfflineIMAP 的配置更轻量、更简单。该命令是由 isync 包提供的。
|
||||||
|
|
||||||
|
配置文件是 `~/.mbsyncrc`。下面是我的例子与一些个人设置。
|
||||||
|
|
||||||
|
```
|
||||||
|
IMAPAccount copyninja
|
||||||
|
Host imap.copyninja.info
|
||||||
|
User vasudev
|
||||||
|
PassCmd "gpg -q --for-your-eyes-only --no-tty --exit-on-status-write-error --batch --passphrase-file ~/path/to/passphrase.txt -d ~/path/to/mailpass.gpg"
|
||||||
|
SSLType IMAPS
|
||||||
|
SSLVersion TLSv1.2
|
||||||
|
CertificateFile /etc/ssl/certs/ca-certificates.crt
|
||||||
|
|
||||||
|
|
||||||
|
IMAPAccount gmail-kamathvasudev
|
||||||
|
Host imap.gmail.com
|
||||||
|
User kamathvasudev@gmail.com
|
||||||
|
PassCmd "gpg -q --for-your-eyes-only --no-tty --exit-on-status-write-error --batch --passphrase-file ~/path/to/passphrase.txt -d ~/path/to/mailpass.gpg"
|
||||||
|
SSLType IMAPS
|
||||||
|
SSLVersion TLSv1.2
|
||||||
|
CertificateFile /etc/ssl/certs/ca-certificates.crt
|
||||||
|
|
||||||
|
IMAPStore copyninja-remote
|
||||||
|
Account copyninja
|
||||||
|
|
||||||
|
IMAPStore gmail-kamathvasudev-remote
|
||||||
|
Account gmail-kamathvasudev
|
||||||
|
|
||||||
|
MaildirStore copyninja-local
|
||||||
|
Path ~/Mail/vasudev-copyninja.info/
|
||||||
|
Inbox ~/Mail/vasudev-copyninja.info/INBOX
|
||||||
|
|
||||||
|
MaildirStore gmail-kamathvasudev-local
|
||||||
|
Path ~/Mail/Gmail-1/
|
||||||
|
Inbox ~/Mail/Gmail-1/INBOX
|
||||||
|
|
||||||
|
Channel copyninja
|
||||||
|
Master :copyninja-remote:
|
||||||
|
Slave :copyninja-local:
|
||||||
|
Patterns *
|
||||||
|
Create Both
|
||||||
|
SyncState *
|
||||||
|
Sync All
|
||||||
|
|
||||||
|
Channel gmail-kamathvasudev
|
||||||
|
Master :gmail-kamathvasudev-remote:
|
||||||
|
Slave :gmail-kamathvasudev-local:
|
||||||
|
# Exclude everything under the internal [Gmail] folder, except the interesting folders
|
||||||
|
Patterns * ![Gmail]*
|
||||||
|
Create Both
|
||||||
|
SyncState *
|
||||||
|
Sync All
|
||||||
|
```
|
||||||
|
|
||||||
|
对上述配置中的一些有趣部分进行一下说明。一个是 PassCmd,它允许你提供 shell 命令来获取帐户的密码。这样可以避免在配置文件中填写密码。我使用 gpg 的对称加密,并在我的磁盘上存储密码。这当然是由 Unix ACL 保护安全的。
|
||||||
|
|
||||||
|
实际上,我想使用我的公钥来加密文件,但当脚本在后台或通过 systemd 运行时,解锁文件看起来很困难 (或者说几乎不可能)。如果你有更好的建议,我洗耳恭听:-)。
|
||||||
|
|
||||||
|
下一个指令部分是 Patterns。这使你可以有选择地同步来自邮件服务器的邮件。这对我来说真的很有帮助,可以排除所有的 “[Gmail]/ folders” 垃圾目录。
|
||||||
|
|
||||||
|
### 邮件分类
|
||||||
|
|
||||||
|
一旦邮件到达你的本地设备,我们需要一种方法来轻松地在邮件读取器中读取邮件。我最初的设置使用本地 dovecot 实例提供同步的 Maildir,并在 Gnus 中阅读。这种设置相比于设置所有的服务器软件是有点大题小作,但 Gnus 无法很好地应付 Maildir 格式,这是最好的方法。这个设置也有一个缺点,那就是在你快速搜索邮件时,要搜索大量邮件。而这就是 notmuch 的用武之地。
|
||||||
|
|
||||||
|
notmuch 允许我轻松索引上千兆字节的邮件档案而找到我需要的东西。我已经创建了一个小脚本,它结合了执行 `mbsync` 和 `notmuch`。我使用 dovecot sieve 来基于实际上创建在服务器端的 Maildirs 标记邮件。下面是我的完整 shell 脚本,它执行同步分类和删除垃圾邮件的任务。
|
||||||
|
|
||||||
|
```
|
||||||
|
#!/bin/sh
|
||||||
|
|
||||||
|
MBSYNC=$(pgrep mbsync)
|
||||||
|
NOTMUCH=$(pgrep notmuch)
|
||||||
|
|
||||||
|
if [ -n "$MBSYNC" -o -n "$NOTMUCH" ]; then
|
||||||
|
echo "Already running one instance of mail-sync. Exiting..."
|
||||||
|
exit 0
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo "Deleting messages tagged as *deleted*"
|
||||||
|
notmuch search --format=text0 --output=files tag:deleted |xargs -0 --no-run-if-empty rm -v
|
||||||
|
|
||||||
|
echo "Moving spam to Spam folder"
|
||||||
|
notmuch search --format=text0 --output=files tag:Spam and \
|
||||||
|
to:vasudev@copyninja.info | \
|
||||||
|
xargs -0 -I {} --no-run-if-empty mv -v {} ~/Mail/vasudev-copyninja.info/Spam/cur
|
||||||
|
notmuch search --format=text0 --output=files tag:Spam and
|
||||||
|
to:vasudev-debian@copyninja.info | \
|
||||||
|
xargs -0 -I {} --no-run-if-empty mv -v {} ~/Mail/vasudev-copyninja.info/Spam/cur
|
||||||
|
|
||||||
|
|
||||||
|
MDIR="vasudev-copyninja.info vasudev-debian Gmail-1"
|
||||||
|
mbsync -Va
|
||||||
|
notmuch new
|
||||||
|
|
||||||
|
for mdir in $MDIR; do
|
||||||
|
echo "Processing $mdir"
|
||||||
|
for fdir in $(ls -d /home/vasudev/Mail/$mdir/*); do
|
||||||
|
if [ $(basename $fdir) != "INBOX" ]; then
|
||||||
|
echo "Tagging for $(basename $fdir)"
|
||||||
|
notmuch tag +$(basename $fdir) -inbox -- folder:$mdir/$(basename $fdir)
|
||||||
|
fi
|
||||||
|
done
|
||||||
|
done
|
||||||
|
```
|
||||||
|
|
||||||
|
因此,在运行 `mbsync` 之前,我搜索所有标记为“deleted”的邮件,并将其从系统中删除。接下来,我在我的帐户上查找标记为“Spam”的邮件,并将其移动到“Spam”文件夹。你没看错,这些邮件逃脱了垃圾邮件过滤器进入到我的收件箱,并被我亲自标记为垃圾邮件。
|
||||||
|
|
||||||
|
运行 `mbsync` 后,我基于它们的文件夹标记邮件(搜索字符串 `folder:`)。这让我可以很容易地得到一个邮件列表的内容,而不需要记住列表地址。
|
||||||
|
|
||||||
|
### 阅读邮件
|
||||||
|
|
||||||
|
现在,我们已经实现同步和分类邮件,是时候来设置阅读部分。我使用 notmuch-emacs 界面来阅读邮件。我使用 emacs 的 Spacemacs 风格,所以我花了一些时间写了一个私有层,它将我所有的快捷键和分类集中在一个地方,而不会扰乱我的整个 `.spacemacs` 文件。你可以在 [notmuch-emacs-layer 仓库][1] 找到我的私有层的代码。
|
||||||
|
|
||||||
|
### 发送邮件
|
||||||
|
|
||||||
|
能阅读邮件这还不够,我们也需要能够回复邮件。而这是最近是我感到迷茫的一个略显棘手的部分,以至于不得不写这篇文章,这样我就不会再忘记了。(当然也不必在网络上参考一些过时的帖子。)
|
||||||
|
|
||||||
|
我的系统发送邮件使用 Postfix 作为 SMTP 客户端,使用我自己的 SMTP 服务器作为它的中继主机。中继的问题是,它不能是具有动态 IP 的主机。有两种方法可以允许具有动态 IP 的主机使用中继服务器, 一种是将邮件来源的 IP 地址放入 `my_network` 或第二个使用 SASL 身份验证。
|
||||||
|
|
||||||
|
我的首选方法是使用 SASL 身份验证。为此,我首先要为每台机器创建一个单独的账户,它将把邮件中继到我的主服务器上。想法是不使用我的主帐户 SASL 进行身份验证。(最初我使用的是主账户,但 Jonas 给出了可行的按账户的想法)
|
||||||
|
|
||||||
|
```
|
||||||
|
adduser <hostname>_relay
|
||||||
|
```
|
||||||
|
|
||||||
|
这里替换 `<hostname>` 为你的笔记本电脑的名称或任何你正在使用的设备。现在我们需要调整 Postfix 作为中继服务器。因此,在 Postfix 配置中添加以下行:
|
||||||
|
|
||||||
|
```
|
||||||
|
# SASL authentication
|
||||||
|
smtp_sasl_auth_enable = yes
|
||||||
|
smtp_tls_security_level = encrypt
|
||||||
|
smtp_sasl_tls_security_options = noanonymous
|
||||||
|
relayhost = [smtp.copyninja.info]:submission
|
||||||
|
smtp_sasl_password_maps = hash:/etc/postfix/sasl_passwd
|
||||||
|
```
|
||||||
|
|
||||||
|
因此, 这里的 `relayhost` 是用于将邮件转发到互联网的 Postfix 实例的服务器名称。`submission` 的部分 Postfix 将邮件转发到端口 587(安全端口)。`smtp_sasl_tls_security_options` 设置为不允许匿名连接。这是必须的,以便中继服务器信任你的移动主机,并同意为你转发邮件。
|
||||||
|
|
||||||
|
`/etc/postfix/sasl_passwd` 是你需要存储用于服务器 SASL 身份验证的帐户密码的文件。将以下内容放入其中。
|
||||||
|
|
||||||
|
```
|
||||||
|
[smtp.example.com]:submission user:password
|
||||||
|
```
|
||||||
|
|
||||||
|
用你已放入 `relayhost` 配置的 SMTP 服务器名称替换 `smtp.example.com`。用你创建的 `<hostname>_relay` 用户及其密码替换 `user` 和 `passwd`。
|
||||||
|
|
||||||
|
若要保护 `sasl_passwd` 文件,并为 Postfix 创建它的哈希文件,使用以下命令。
|
||||||
|
|
||||||
|
```
|
||||||
|
chown root:root /etc/postfix/sasl_passwd
|
||||||
|
chmod 0600 /etc/postfix/sasl_passwd
|
||||||
|
postmap /etc/postfix/sasl_passwd
|
||||||
|
```
|
||||||
|
|
||||||
|
最后一条命令将创建 `/etc/postfix/sasl_passwd.db` 文件,它是你的文件的 `/etc/postfix/sasl_passwd` 的哈希文件,具有相同的所有者和权限。现在重新加载 Postfix,并使用 `mail` 命令检查邮件是否从你的系统中发出。
|
||||||
|
|
||||||
|
### Bonus 的部分
|
||||||
|
|
||||||
|
好吧,因为我有一个脚本创建以上结合了邮件的同步和分类。我继续创建了一个 systemd 计时器,以定期同步后台的邮件。就我而言,每 10 分钟一次。下面是 `mailsync.timer` 文件。
|
||||||
|
|
||||||
|
```
|
||||||
|
[Unit]
|
||||||
|
Description=Check Mail Every 10 minutes
|
||||||
|
RefuseManualStart=no
|
||||||
|
RefuseManualStop=no
|
||||||
|
|
||||||
|
[Timer]
|
||||||
|
Persistent=false
|
||||||
|
OnBootSec=5min
|
||||||
|
OnUnitActiveSec=10min
|
||||||
|
Unit=mailsync.service
|
||||||
|
|
||||||
|
[Install]
|
||||||
|
WantedBy=default.target
|
||||||
|
```
|
||||||
|
|
||||||
|
下面是 mailsync.service 服务,这是 mailsync.timer 执行我们的脚本所需要的。
|
||||||
|
|
||||||
|
```
|
||||||
|
[Unit]
|
||||||
|
Description=Check Mail
|
||||||
|
RefuseManualStart=no
|
||||||
|
RefuseManualStop=yes
|
||||||
|
|
||||||
|
[Service]
|
||||||
|
Type=oneshot
|
||||||
|
ExecStart=/usr/local/bin/mail-sync
|
||||||
|
StandardOutput=syslog
|
||||||
|
StandardError=syslog
|
||||||
|
```
|
||||||
|
|
||||||
|
将这些文件置于 `/etc/systemd/user` 目录下并运行以下代码去开启它们:
|
||||||
|
|
||||||
|
```
|
||||||
|
systemctl enable --user mailsync.timer
|
||||||
|
systemctl enable --user mailsync.service
|
||||||
|
systemctl start --user mailsync.timer
|
||||||
|
```
|
||||||
|
|
||||||
|
这就是我从系统同步和发送邮件的方式。我从 Jonas Smedegaard 那里了解到了 afew,他审阅了这篇帖子。因此, 下一步, 我将尝试使用 afew 改进我的 notmuch 配置,当然还会有一个后续的帖子:-)。
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: https://copyninja.info/blog/email_setup.html
|
||||||
|
|
||||||
|
作者:[copyninja][a]
|
||||||
|
译者:[lixinyuxx](https://github.com/lixinyuxx)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]:https://copyninja.info
|
||||||
|
[1]:https://source.copyninja.info/notmuch-emacs-layer.git/
|
@ -0,0 +1,91 @@
|
|||||||
|
27 个全方位的开放式教育解决方案
|
||||||
|
======
|
||||||
|
|
||||||
|
> 阅读这些 2017 年 Opensource.com 发布的开放如何改进教育和学习的好文章。
|
||||||
|
|
||||||
|
![27 open solutions to everything in education](https://opensource.com/sites/default/files/styles/image-full-size/public/lead-images/EDU_OpenEducationResources_520x292_cm.png?itok=9y4FGgRo)
|
||||||
|
|
||||||
|
开放式理念 (从开源软件到开放硬件,再到开放原则) 正在改变教育的范式。因此,为了庆祝今年发生的一切,我收集了 2017 年(译注:本文原发布于 2018 年初)在 Opensource.com 上发表的 27 篇关于这个主题的最好的文章。我把它们分成明确的主题,而不是按人气来分类。而且,如果这 27 个故事不能满足你对教育方面开源信息的胃口,那就看看我们的合作文章吧 “[教育如何借助 Linux 和树莓派][30]”。
|
||||||
|
|
||||||
|
### 开放对每个人都有好处
|
||||||
|
|
||||||
|
1. [书评:《OPEN》探讨了开放性的广泛文化含义][1]:Scott Nesbitt 评价 David Price 的书 《OPEN》 ,该书探讨了 “开放” 不仅仅是技术转变的观点,而是 “我们未来将如何工作、生活和学习”。
|
||||||
|
2. [通过开源技能快速开始您的职业生涯][2]: VM (Vicky) Brasseur 指出了如何借助学习开源在工作群体中脱颖而出。这个建议不仅仅是针对程序员的;设计师、作家、营销人员和其他创意专业人士也对开源的成功至关重要。
|
||||||
|
3. [研究生学位可以让你跳槽到开源职位][3]:引用的研究表明会 Linux 技能会带来更高的薪水, Joshua Pearce 说对开源的熟练和研究生学位是无与伦比的职业技能组合。
|
||||||
|
4. [彻底改变了宾夕法尼亚的学校文化的三种实践][4]:Charlie Reisinger 向我们展示了开放式实践是如何在宾夕法尼亚州的一个学区创造一种更具包容性、敏捷性和开放性的文化的。Charlie 说,这不仅仅是为了省钱;该区还受益于 “开放式领导原则,促进师生创新,帮助更好地吸引社区,创造一个更有活力和包容性的学习社区”。
|
||||||
|
5. [使用开源工具促使学生进步的 15 种方法][5]:我写了开源是如何让学生自由探索、补拙和学习的,不管他们是在学习基本的数字化素养,还是通过有趣的项目来扩展这些技能。
|
||||||
|
6. [开发人员有机会编写好的代码][6]:开源往往是对社会有益的项目的支柱。正如 Benetech Labs 副总裁 Ahn Bui 在这次采访中指出的那样:“建立开放数据标准是打破数据孤岛不可或缺的一步。这些开放标准将为互操作性提供基础,进而转化为更多的组织共同建设,往往更具成本效益。最终目标是以同样的成本甚至更低的成本为更多的人服务。”
|
||||||
|
|
||||||
|
### 用于再融合和再利用的开放式教育资源
|
||||||
|
|
||||||
|
1. [学术教员可以和维基百科一起教学吗?][7]:Wiki Ed 的项目总监 LiAnna Davis 讨论<ruby>开放式教育资源<rt>open educational resources</rt></ruby> (OER) ,如 Wiki Ed,是如何提供高质量且经济实惠的开源学习资源作为课堂教学工具。
|
||||||
|
2. [书本内外?开放教育资源的状态][8]:Cable Green 是 Creative Common 开放教育主管,分享了高等教育中教育面貌是如何变化的,以及 Creative Common 正在采取哪些措施来促进教育。
|
||||||
|
3. [急需符合标准的课程的学校系统找到了希望][9]:Karen Vaites 是 Open Up Resources 社区布道师和首席营销官,谈论了非营利组织努力为 K-12 学校提供开放的、标准一致的课程。
|
||||||
|
4. [夏威夷大学如何解决当今高等教育的问题][10]:夏威夷大学 Manoa 分校的教育技术专家 Billy Meinke 表示,在大学课程中过渡到 ORE 将 “使教师能够控制他们教授的内容,我们预计这将为他们节省学生的费用。”
|
||||||
|
5. [开放式课程如何削减高等教育成本][11]:塞勒学院的教育总监 Devon Ritter 报告了塞勒学院是如何建立以公开许可内容为基础的大学学分课程,目的是使更多的人能够负担得起和获得高等教育。
|
||||||
|
6. [开放教育资源运动在提速][12]:Alexis Clifton 是纽约州立大学的 OER 服务的执行董事,描述了纽约 800 万美元的投资如何刺激开放教育的增长,并使大学更实惠。
|
||||||
|
7. [开放项目合作,从小学到大学教室][13]:来自杜克大学的 Aria F. Chernik 探索 OSPRI (开源教育学的研究与创新), 这是杜克大学和红帽的合作,旨在建立一个 21 世纪的,开放设计的 preK-12 学习生态系统。
|
||||||
|
8. [Perma.cc 将阻止学术链接腐烂][14]::弗吉尼亚理工大学的 Phillip Young 写的关于 Perma.cc 的文章,这种一种“链接腐烂”的解决方案,在学术论文中的超链接随着时间的推移而消失或变化的概览很高。
|
||||||
|
9. [开放教育:学生如何通过创建开放教科书来节省资金][15]:OER 先驱 Robin DeRosa 谈到 “引入公开许可教科书的自由,以及教育和学习应结合包容性生态系统,以增强公益的总体理念”。
|
||||||
|
|
||||||
|
### 课堂上的开源工具
|
||||||
|
|
||||||
|
1. [开源棋盘游戏如何拯救地球][16]:Joshua Pearce 写的关于拯救地球的一个棋盘游戏,这是一款让学生在玩乐和为创客社区做出贡献的同时解决环境问题的棋盘游戏。
|
||||||
|
2. [一个教孩子们如何阅读的新 Android 应用程序][17]:Michael Hall 谈到了他在儿子被诊断为自闭症后为他开发的儿童识字应用 Phoenicia,以及良好编码的价值,和为什么用户测试比你想象的更重要。
|
||||||
|
3. [8 个用于教育的开源 Android 应用程序][18]:Joshua Allen Holm 推荐了 8 个来自 F-Droid 软件库的开源应用,使我们可以将智能手机用作学习工具。
|
||||||
|
4. [MATLA B的 3 种开源替代方案][19]:Jason Baker 更新了他 2016 年的开源数学计算软件调查报告,提供了 MATLAB 的替代方案,这是数学、物理科学、工程和经济学中几乎无处不在的昂贵的专用解决方案。
|
||||||
|
5. [SVG 与教孩子编码有什么关系?][20]:退休工程师 Jay Nick 谈论他如何使用艺术作为一种创新的方式,向学生介绍编码。他在学校做志愿者,使用 SVG 来教授一种结合数学和艺术原理的编码方法。
|
||||||
|
6. [5 个破灭的神话:在高等教育中使用开源][21]: 拥有德克萨斯理工大学美术博士学位的 Kyle Conway 分享他在一个由专有解决方案统治的世界中使用开源工具的经验。 Kyle 说有一种偏见,反对在计算机科学以外的学科中使用开源:“很多人认为非技术专业的学生不能使用 Linux,他们对在高级学位课程中使用 Linux 的人做出了很多假设……嗯,这是有可能的,我就是证明。”
|
||||||
|
7. [大学开源工具列表][22]:Aaron Cocker 概述了他在攻读计算机科学本科学位时使用的开源工具 (包括演示、备份和编程软件)。
|
||||||
|
8. [5 个可帮助您学习优秀 KDE 应用程序][23]:Zsolt Szakács 提供五个 KDE 应用程序,可以帮助任何想要学习新技能或培养现有技能的人。
|
||||||
|
|
||||||
|
### 在教室编码
|
||||||
|
|
||||||
|
1. [如何尽早让下一代编码][24]:Bryson Payne 说我们需要在高中前教孩子们学会编码: 到了九年级,80% 的女孩和 60% 的男孩已经从 STEM 职业中自选。但他建议,这不仅仅是就业和缩小 IT 技能差距的问题。“教一个年轻人编写代码可能是你能给他们的最改变生活的技能。而且这不仅仅是一个职业提升者。编码是关于解决问题,它是关于创造力,更重要的是,它是提升能力”。
|
||||||
|
2. [孩子们无法在没有计算机的情况下编码][25]:Patrick Masson 推出了 FLOSS 儿童桌面计划, 该计划教授服务不足学校的学生使用开源软件 (如 Linux、LibreOffice 和 GIMP) 重新利用较旧的计算机。该计划不仅为破旧或退役的硬件注入新的生命,还为学生提供了重要的技能,而且还为学生提供了可能转化为未来职业生涯的重要技能。
|
||||||
|
3. [如今 Scratch 是否能像 80 年代的 LOGO 语言一样教孩子们编码?][26]:Anderson Silva 提出使用 [Scratch][27] 以激发孩子们对编程的兴趣,就像在 20 世纪 80 年代开始使用 LOGO 语言时一样。
|
||||||
|
4. [通过这个拖放框架学习Android开发][28]:Eric Eslinger 介绍了 App Inventor,这是一个编程框架,用于构建 Android 应用程序使用可视块语言(类似 Scratch 或者 [Snap][29])。
|
||||||
|
|
||||||
|
在这一年里,我们了解到,教育领域的各个方面都有了开放的解决方案,我预计这一主题将在 2018 年及以后继续下去。在未来的一年里,你是否希望 Opensource.com 涵盖开放式的教育主题?如果是, 请在评论中分享你的想法。
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: https://opensource.com/article/18/1/best-open-education
|
||||||
|
|
||||||
|
作者:[Don Watkins][a]
|
||||||
|
译者:[lixinyuxx](https://github.com/lixinyuxx)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]:https://opensource.com/users/don-watkins
|
||||||
|
[1]:https://opensource.com/article/17/7/book-review-open
|
||||||
|
[2]:https://opensource.com/article/17/8/jump-start-your-career
|
||||||
|
[3]:https://opensource.com/article/17/1/grad-school-open-source-academic-lab
|
||||||
|
[4]:https://opensource.com/article/17/7/open-school-leadership
|
||||||
|
[5]:https://opensource.com/article/17/7/empower-students-open-source-tools
|
||||||
|
[6]:https://opensource.com/article/17/3/interview-anh-bui-benetech-labs
|
||||||
|
[7]:https://opensource.com/article/17/1/Wiki-Education-Foundation
|
||||||
|
[8]:https://opensource.com/article/17/2/future-textbooks-cable-green-creative-commons
|
||||||
|
[9]:https://opensource.com/article/17/1/open-up-resources
|
||||||
|
[10]:https://opensource.com/article/17/2/interview-education-billy-meinke
|
||||||
|
[11]:https://opensource.com/article/17/7/college-alternatives
|
||||||
|
[12]:https://opensource.com/article/17/10/open-educational-resources-alexis-clifton
|
||||||
|
[13]:https://opensource.com/article/17/3/education-should-be-open-design
|
||||||
|
[14]:https://opensource.com/article/17/9/stop-link-rot-permacc
|
||||||
|
[15]:https://opensource.com/article/17/11/creating-open-textbooks
|
||||||
|
[16]:https://opensource.com/article/17/7/save-planet-board-game
|
||||||
|
[17]:https://opensource.com/article/17/4/phoenicia-education-software
|
||||||
|
[18]:https://opensource.com/article/17/8/8-open-source-android-apps-education
|
||||||
|
[19]:https://opensource.com/alternatives/matlab
|
||||||
|
[20]:https://opensource.com/article/17/5/coding-scalable-vector-graphics-make-steam
|
||||||
|
[21]:https://opensource.com/article/17/5/how-linux-higher-education
|
||||||
|
[22]:https://opensource.com/article/17/6/open-source-tools-university-student
|
||||||
|
[23]:https://opensource.com/article/17/6/kde-education-software
|
||||||
|
[24]:https://opensource.com/article/17/8/teach-kid-code-change-life
|
||||||
|
[25]:https://opensource.com/article/17/9/floss-desktops-kids
|
||||||
|
[26]:https://opensource.com/article/17/3/logo-scratch-teach-programming-kids
|
||||||
|
[27]:https://scratch.mit.edu/
|
||||||
|
[28]:https://opensource.com/article/17/8/app-inventor-android-app-development
|
||||||
|
[29]:http://snap.berkeley.edu/
|
||||||
|
[30]:https://opensource.com/article/17/12/best-opensourcecom-linux-and-raspberry-pi-education
|
95
published/201812/20180128 Getting Linux Jobs.md
Normal file
95
published/201812/20180128 Getting Linux Jobs.md
Normal file
@ -0,0 +1,95 @@
|
|||||||
|
Linux 求职建议
|
||||||
|
======
|
||||||
|
|
||||||
|
通过对招聘网站数据的仔细研究,我们发现,即使是非常有经验的 Linux 程序员,也会在面试中陷入困境。
|
||||||
|
|
||||||
|
这就导致了很多优秀并且有经验的人无缘无故地找不到合适的工作,因为如今的就业市场需要我们有一些手段来提高自己的竞争力。
|
||||||
|
|
||||||
|
我有两个同事和一个表哥,他们都有 RedHat 认证,管理过比较大的服务器机房,也都收到过前雇主的认真推荐。
|
||||||
|
|
||||||
|
可是,在他们应聘的时候,所有的这些证书、本身的能力、工作经验好像都没有起到任何作用,他们所面对的招聘广告是某人从技术词汇中临时挑选的一些“技能片段”所组成的。
|
||||||
|
|
||||||
|
现如今,礼貌变得过时了,**不回应**变成了发布招聘广告的公司的新沟通方式。
|
||||||
|
|
||||||
|
这同样也意味着大多公司的招聘或者人事可能会**错过**非常优秀的应聘者。
|
||||||
|
|
||||||
|
我之所以敢说的如此肯定,是因为现在招聘广告大多数看上去都非常的滑稽。
|
||||||
|
|
||||||
|
[Reallylinux.com][3] 另一位特约撰稿人 Walter ,发表过一篇关于 [招聘广告疯掉了][4] 的文章。
|
||||||
|
|
||||||
|
他说的也许是对的,可是我认为 Linux 工作应聘者可以通过注意招聘广告的**三个关键点**避免落入陷阱。
|
||||||
|
|
||||||
|
**首先**,很少会有 Linux 系统管理员的招聘广告只针对 Linux 有要求。
|
||||||
|
|
||||||
|
一定要注意很少有 Linux 系统管理员的职位是实际在服务器上跑 Linux的,反而,很多在搜索 “Linux 管理员” 得到的职位实际上是指大量的 *NX 操作系统的。
|
||||||
|
|
||||||
|
举个例子,有一则关于 **Linux 管理员** 的招聘广告:
|
||||||
|
|
||||||
|
> 该职位需要为建立系统集成提供支持,尤其是 BSD 应用的系统安装...
|
||||||
|
|
||||||
|
或者有一些其他的要求:
|
||||||
|
|
||||||
|
> 有 Windows 系统管理经验的。
|
||||||
|
|
||||||
|
最为讽刺的是,如果你在应聘面试的时候表现出专注于 Linux 的话,你可能不会被聘用。
|
||||||
|
|
||||||
|
另外,如果你直接把 Linux 写在你的特长或者专业上,他们可能都不会仔细看你的简历,因为他们根本区分不了 UNIX、BSD、Linux。
|
||||||
|
|
||||||
|
最终的结果就是,如果你太老实,只在简历上写了 Linux,你可能会被直接过掉,但是如果你把 Linux 改成 UNIX/Linux 的话,可能会走得更远。
|
||||||
|
|
||||||
|
我有两个同事最后修改了他们的简历,然后获得了更好的面试机会,虽然依旧没有被聘用,因为大多数招聘广告其实已经内定人员了,这些招聘信息被放出来仅仅是为了表现出他们有招聘的想法。
|
||||||
|
|
||||||
|
**第二点**,公司里唯一在乎系统管理员职位的只有技术主管,其他人包括人事或管理层根本不关心这个。
|
||||||
|
|
||||||
|
我记得有一次开会旁听的时候,听见一个执行副总裁把服务器管理人员说成“一毛钱一打的人”,这种想法是多么的奇怪啊。
|
||||||
|
|
||||||
|
讽刺的是,等到邮件系统出故障,电话交换机连接时不时会断开,或者核心商业文件从企业内网中消失的时候,这些总裁又是最先打电话给系统管理员的。
|
||||||
|
|
||||||
|
或许如果他们不整天在电话留言中说那么多空话,或者不往邮件里塞满妻子的照片和旅行途中的照片的话,服务器可能就不会崩溃。
|
||||||
|
|
||||||
|
请注意,招聘 Linux 运维或者服务器管理员的广告被放出来是因为公司**技术层**认为有迫切的需求。你也不需要和人事或者公司高层聊什么,搞清楚谁是招聘的技术经理然后打电话给他们。
|
||||||
|
|
||||||
|
你需要直接联系他们因为“有些技术问题”是人事回答不了的,即使你只有 60 秒的时间可以和他们交流,你也必须抓住这个机会和真正有需求并且懂技术的人沟通。
|
||||||
|
|
||||||
|
那如果人事的漂亮 MM 不让你直接联系技术怎么办呢?
|
||||||
|
|
||||||
|
开始记得问人事一些技术性问题,比如说他们的 Linux 集群是如何建立的,它们运行在独立的虚拟机上吗?这些技术性的问题会让人事变得不耐烦,最后让你有机会问出“我能不能直接联系你们团队的技术人员”。
|
||||||
|
|
||||||
|
如果对方的回答是“应该可以”或者“稍后回复你”,那么他们可能已经在两周前就已经计划好了找一个人来填补这个空缺,比如说人事部员工的未婚夫。**他们只是不希望看起来太像裙带主义,而是带有一点利己主义的不确定主义。**
|
||||||
|
|
||||||
|
所以一定要记得花点时间弄清楚到底谁是发布招聘广告的直接**技术**负责人,然后和他们聊一聊,这可能会让你少一番胡扯并且让你更有可能应聘成功。
|
||||||
|
|
||||||
|
**第三点**,现在的招聘广告很少有完全真实的内容了。
|
||||||
|
|
||||||
|
我以前见过一个招聘具有高级别专家也不会有的专门知识的初级系统管理员的广告,他们的计划是列出公司的发展计划蓝图,然后找到应聘者。
|
||||||
|
|
||||||
|
在这种情况下,你应聘 Linux 管理员职位应该提供几个关键性信息,例如工作经验和相关证书。
|
||||||
|
|
||||||
|
诀窍在于,用这些关键词尽量装点你的简历,以匹配他们的招聘信息,这样他们几乎不可能发现你缺失了哪个关键词。
|
||||||
|
|
||||||
|
这并不一定会让你成功找到一份工作,但它可以让你获得一次面试机会,这也算是一个巨大的进步。
|
||||||
|
|
||||||
|
通过理解和应用以上三点,或许可以让那些寻求 Linux 管理员工作的人能够比那些只有一线地狱机会的人领先一步。
|
||||||
|
|
||||||
|
即使这些建议不能让你马上得到面试机会,你也可以利用这些经验和意识去参加贸易展或公司主办的技术会议等活动。
|
||||||
|
|
||||||
|
我强烈建议你们也经常参加这种活动,尤其是当它们比较近的话,可以给你一个扩展人脉的机会。
|
||||||
|
|
||||||
|
请记住,如今的职业人脉已经失去了原来的意义了,现在只是可以用来获取“哪些公司实际上在招聘、哪些公司只是为了给股东带来增长的表象而在职位方面撒谎”的小道消息。
|
||||||
|
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: http://reallylinux.com/docs/gettinglinuxjobs.shtml
|
||||||
|
|
||||||
|
作者:[Andrea W.Codingly][a]
|
||||||
|
译者:[Ryze-Borgia](https://github.com/Ryze-Borgia)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]:http://reallylinux.com
|
||||||
|
[1]:http://www.reallylinux.com
|
||||||
|
[2]:http://reallylinux.com/docs/linuxrecessionproof.shtml
|
||||||
|
[3]:http://reallylinux.com
|
||||||
|
[4]:http://reallylinux.com/docs/wantadsmad.shtml
|
@ -0,0 +1,179 @@
|
|||||||
|
用于游戏开发的图形和音乐工具
|
||||||
|
======
|
||||||
|
> 要在三天内打造一个可玩的游戏,你需要一些快速而稳定的好工具。
|
||||||
|
|
||||||
|
![](https://opensource.com/sites/default/files/styles/image-full-size/public/lead-images/OSDC_Life_opengame.png?itok=JPxruL3k)
|
||||||
|
|
||||||
|
在十月初,我们的俱乐部马歇尔大学的 [Geeks and Gadgets][1] 参加了首次 [Open Jam][2],这是一个庆祝最佳开源工具的游戏 Jam。游戏 Jam 是一种活动,参与者以团队协作的方式来开发有趣的计算机游戏。Jam 一般都很短,仅有三天,并且非常累。Opensource.com 在八月下旬[发布了][3] Open Jam 活动,足有 [45 支游戏][4] 进入到了竞赛中。
|
||||||
|
|
||||||
|
我们的俱乐部希望在我们的项目中创建和使用开放源码软件,所以 Open Jam 自然是我们想要参与的 Jam 了。我们提交的游戏是一个实验性的游戏,名为 [Mark My Words][5]。我们使用了多种自由和开放源码 (FOSS) 工具来开发它;在这篇文章中,我们将讨论一些我们使用的工具和我们注意到可能有潜在阻碍的地方。
|
||||||
|
|
||||||
|
### 音频工具
|
||||||
|
|
||||||
|
#### MilkyTracker
|
||||||
|
|
||||||
|
[MilkyTracker][6] 是一个可用于编曲老式视频游戏中的音乐的软件包。它是一种<ruby>[音乐声道器][7]<rt>music tracker</rt></ruby>,是一个强大的 MOD 和 XM 文件创建器,带有基于特征网格的模式编辑器。在我们的游戏中,我们使用它来编曲大多数的音乐片段。这个程序最好的地方是,它比我们其它的大多数工具消耗更少的硬盘空间和内存。虽然如此,MilkyTracker 仍然非常强大。
|
||||||
|
|
||||||
|
![](https://opensource.com/sites/default/files/u128651/mtracker.png)
|
||||||
|
|
||||||
|
其用户界面需要一会来习惯,这里有对一些想试用 MilkyTracker 的音乐家的一些提示:
|
||||||
|
|
||||||
|
* 转到 “Config > Misc.” ,设置编辑模式的控制风格为 “MilkyTracker”,这将给你提供几乎全部现代键盘快捷方式。
|
||||||
|
* 用 `Ctrl+Z` 撤销
|
||||||
|
* 用 `Ctrl+Y` 重做
|
||||||
|
* 用空格键切换模式编辑方式
|
||||||
|
* 用退格键删除先前的音符
|
||||||
|
* 用插入键来插入一行
|
||||||
|
* 默认情况下,一个音符将持续作用,直到它在该频道上被替换。你可以明确地结束一个音符,通过使用一个反引号(`)键来插入一个 KeyOff 音符
|
||||||
|
* 在你开始谱写乐曲前,你需要创建或查找采样。我们建议在诸如 [Freesound][9] 或 [ccMixter][10] 这样的网站上查找采用 [Creative Commons][8] 协议的采样,
|
||||||
|
|
||||||
|
另外,把 [MilkyTracker 文档页面][11] 放在手边。它含有数不清的教程和手册的链接。一个好的起点是在该项目 wiki 上的 [MilkyTracker 指南][12]。
|
||||||
|
|
||||||
|
#### LMMS
|
||||||
|
|
||||||
|
我们的两个音乐家使用多用途的现代音乐创建工具 [LMMS][13]。它带有一个绝妙的采样和效果库,以及多种多样的灵活的插件来生成独特的声音。LMMS 的学习曲线令人吃惊的低,在某种程度上是因为其好用的节拍/低音线编辑器。
|
||||||
|
|
||||||
|
![](https://opensource.com/sites/default/files/u128651/lmms_plugins.png)
|
||||||
|
|
||||||
|
我们对于想试试 LMMS 的音乐家有一个建议:使用插件。对于 [chiptune][14]式音乐,我们推荐 [sfxr][15]、[BitInvader][16] 和 [FreeBoy][17]。对于其它风格,[ZynAddSubFX][18] 是一个好的选择。它配备了各种合成仪器,可以根据您的需要进行更改。
|
||||||
|
|
||||||
|
### 图形工具
|
||||||
|
|
||||||
|
#### Tiled
|
||||||
|
|
||||||
|
在开放源码游戏开发中,[Tiled][19] 是一个流行的贴片地图编辑器。我们使用它为来为我们在游戏场景中组合连续的、复古式的背景。
|
||||||
|
|
||||||
|
![](https://opensource.com/sites/default/files/u128651/tiled.png)
|
||||||
|
|
||||||
|
Tiled 可以导出地图为 XML、JSON 或普通的图片。它是稳定的、跨平台的。
|
||||||
|
|
||||||
|
Tiled 的功能之一允许你在地图上定义和放置随意的游戏对象,例如硬币和提升道具,但在 jam 期间我们没有使用它。你需要做的全部是以贴片集的方式加载对象的图像,然后使用“插入平铺”来放置它们。
|
||||||
|
|
||||||
|
一般来说,对于需要一个地图编辑器的项目,Tiled 是我们所推荐的软件中一个不可或缺的部分。
|
||||||
|
|
||||||
|
#### Piskel
|
||||||
|
|
||||||
|
[Piskel][20] 是一个像素艺术编辑器,它的源文件代码以 [Apache 2.0 协议][21] 发布。在这次 Jam 期间,们的大多数的图像资源都使用 Piskel 来处理,我们当然也将在未来的工程中使用它。
|
||||||
|
|
||||||
|
在这个 Jam 期间,Piskel 极大地帮助我们的两个功能是<ruby>洋葱皮<rt>Onion skin</rt></ruby>和<ruby>精灵序列图<rt>spritesheet</rt></ruby>导出。
|
||||||
|
|
||||||
|
##### 洋葱皮
|
||||||
|
|
||||||
|
洋葱皮功能将使 Piskel 以虚影显示你编辑的动画的前一帧和后一帧的,像这样:
|
||||||
|
|
||||||
|
![](https://opensource.com/sites/default/files/u128651/onionshow.gif)
|
||||||
|
|
||||||
|
洋葱皮是很方便的,因为它适合作为一个绘制指引和帮助你在整个动画进程中保持角色的一致形状和体积。 要启用它,只需单击屏幕右上角预览窗口下方的洋葱形图标即可。
|
||||||
|
|
||||||
|
![](https://opensource.com/sites/default/files/u128651/onionenable.png)
|
||||||
|
|
||||||
|
##### 精灵序列图导出
|
||||||
|
|
||||||
|
Piskel 将动画导出为精灵序列图的能力也非常有用。精灵序列图是一个包含动画所有帧的光栅图像。例如,这是我们从 Piskel 导出的精灵序列图:
|
||||||
|
|
||||||
|
![](https://opensource.com/sites/default/files/u128651/sprite-artist.png)
|
||||||
|
|
||||||
|
该精灵序列图包含两帧。一帧位于图像的上半部分,另一帧位于图像的下半部分。精灵序列图通过从单个文件加载整个动画,大大简化了游戏的代码。这是上面精灵序列图的动画版本:
|
||||||
|
|
||||||
|
![](https://opensource.com/sites/default/files/u128651/sprite-artist-anim.gif)
|
||||||
|
|
||||||
|
##### Unpiskel.py
|
||||||
|
|
||||||
|
在 Jam 期间,我们很多次想批量转换 Piskel 文件到 PNG 文件。由于 Piskel 文件格式基于 JSON,我们写一个基于 GPLv3 协议的名为 [unpiskel.py][22] 的 Python 小脚本来做转换。
|
||||||
|
|
||||||
|
它像这样被调用的:
|
||||||
|
|
||||||
|
```
|
||||||
|
python unpiskel.py input.piskel
|
||||||
|
```
|
||||||
|
|
||||||
|
这个脚本将从一个 Piskel 文件(这里是 `input.piskel`)中提取 PNG 数据帧和图层,并将它们各自存储。这些文件采用模式 `NAME_XX_YY.png` 命名,在这里 `NAME` 是 Piskel 文件的缩减名称,`XX` 是帧的编号,`YY` 是层的编号。
|
||||||
|
|
||||||
|
因为脚本可以从一个 shell 中调用,它可以用在整个文件列表中。
|
||||||
|
|
||||||
|
```
|
||||||
|
for f in *.piskel; do python unpiskel.py "$f"; done
|
||||||
|
```
|
||||||
|
|
||||||
|
### Python、Pygame 和 cx_Freeze
|
||||||
|
|
||||||
|
#### Python 和 Pygame
|
||||||
|
|
||||||
|
我们使用 [Python][23] 语言来制作我们的游戏。它是一个脚本语言,通常被用于文本处理和桌面应用程序开发。它也可以用于游戏开发,例如像 [Angry Drunken Dwarves][24] 和 [Ren'Py][25] 这样的项目所展示的。这两个项目都使用一个称为 [Pygame][26] 的 Python 库来显示图形和产生声音,所以我们也决定在 Open Jam 中使用这个库。
|
||||||
|
|
||||||
|
Pygame 被证明是既稳定又富有特色,并且它对我们创建的街机式游戏来说是很棒的。在低分辨率时,库的速度足够快的,但是在高分辨率时,它只用 CPU 的渲染开始变慢。这是因为 Pygame 不使用硬件加速渲染。然而,开发者可以充分利用 OpenGL 基础设施的优势。
|
||||||
|
|
||||||
|
如果你正在寻找一个好的 2D 游戏编程库,Pygame 是值得密切注意的一个。它的网站有 [一个好的教程][27] 可以作为起步。务必看看它!
|
||||||
|
|
||||||
|
#### cx_Freeze
|
||||||
|
|
||||||
|
准备发行我们的游戏是有趣的。我们知道,Windows 用户不喜欢装一套 Python,并且要求他们来安装它可能很过分。除此之外,他们也可能必须安装 Pygame,在 Windows 上,这不是一个简单的工作。
|
||||||
|
|
||||||
|
很显然:我们必须放置我们的游戏到一个更方便的格式中。很多其他的 Open Jam 参与者使用专有的游戏引擎 Unity,它能够使他们的游戏在网页浏览器中来玩。这使得它们非常方便地来玩。便利性是一个我们的游戏中根本不存在的东西。但是,感谢生机勃勃的 Python 生态系统,我们有选择。已有的工具可以帮助 Python 程序员将他们的游戏做成 Windows 上的发布版本。我们考虑过的两个工具是 [cx_Freeze][28] 和 [Pygame2exe][29](它使用 [py2exe][30])。我们最终决定用 cx_Freeze,因为它是跨平台的。
|
||||||
|
|
||||||
|
在 cx_Freeze 中,你可以把一个单脚本游戏打包成发布版本,只要在 shell 中运行一个命令,像这样:
|
||||||
|
|
||||||
|
```
|
||||||
|
cxfreeze main.py --target-dir dist
|
||||||
|
```
|
||||||
|
|
||||||
|
`cxfreeze` 的这个调用将把你的脚本(这里是 `main.py`)和在你系统上的 Python 解释器捆绑到到 `dist` 目录。一旦完成,你需要做的是手动复制你的游戏的数据文件到 `dist` 目录。你将看到,`dist` 目录包含一个可以运行来开始你的游戏的可执行文件。
|
||||||
|
|
||||||
|
这里有使用 cx_Freeze 的更复杂的方法,允许你自动地复制数据文件,但是我们发现简单的调用 `cxfreeze` 足够满足我们的需要。感谢这个工具,我们使我们的游戏玩起来更便利一些。
|
||||||
|
|
||||||
|
### 庆祝开源
|
||||||
|
|
||||||
|
Open Jam 是庆祝开源模式的软件开发的重要活动。这是一个分析开源工具的当前状态和我们在未来工作中需求的一个机会。对于游戏开发者探求其工具的使用极限,学习未来游戏开发所必须改进的地方,游戏 Jam 或许是最好的时机。
|
||||||
|
|
||||||
|
开源工具使人们能够在不损害自由的情况下探索自己的创造力,而无需预先投入资金。虽然我们可能不会成为专业的游戏开发者,但我们仍然能够通过我们的简短的实验性游戏 [Mark My Words][5] 获得一点点体验。它是一个以语言学为主题的游戏,描绘了虚构的书写系统在其历史中的演变。还有很多其他不错的作品提交给了 Open Jam,它们都值得一试。 真的,[去看看][31]!
|
||||||
|
|
||||||
|
在本文结束前,我们想要感谢所有的 [参加俱乐部的成员][32],使得这次经历真正的有价值。我们也想要感谢 [Michael Clayton][33]、[Jared Sprague][34] 和 [Opensource.com][35] 主办 Open Jam。简直酷毙了。
|
||||||
|
|
||||||
|
现在,我们对读者提出了一些问题。你是一个 FOSS 游戏开发者吗?你选择的工具是什么?务必在下面留下一个评论!
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: https://opensource.com/article/18/1/graphics-music-tools-game-dev
|
||||||
|
|
||||||
|
作者:[Charlie Murphy][a]
|
||||||
|
译者:[robsean](https://github.com/robsean)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]:https://opensource.com/users/rsg167
|
||||||
|
[1]:http://mugeeks.org/
|
||||||
|
[2]:https://itch.io/jam/open-jam-1
|
||||||
|
[3]:https://opensource.com/article/17/8/open-jam-announcement
|
||||||
|
[4]:https://opensource.com/article/17/11/open-jam
|
||||||
|
[5]:https://mugeeksalpha.itch.io/mark-omy-words
|
||||||
|
[6]:http://milkytracker.titandemo.org/
|
||||||
|
[7]:https://en.wikipedia.org/wiki/Music_tracker
|
||||||
|
[8]:https://creativecommons.org/
|
||||||
|
[9]:https://freesound.org/
|
||||||
|
[10]:http://ccmixter.org/view/media/home
|
||||||
|
[11]:http://milkytracker.titandemo.org/documentation/
|
||||||
|
[12]:https://github.com/milkytracker/MilkyTracker/wiki/MilkyTracker-Guide
|
||||||
|
[13]:https://lmms.io/
|
||||||
|
[14]:https://en.wikipedia.org/wiki/Chiptune
|
||||||
|
[15]:https://github.com/grimfang4/sfxr
|
||||||
|
[16]:https://lmms.io/wiki/index.php?title=BitInvader
|
||||||
|
[17]:https://lmms.io/wiki/index.php?title=FreeBoy
|
||||||
|
[18]:http://zynaddsubfx.sourceforge.net/
|
||||||
|
[19]:http://www.mapeditor.org/
|
||||||
|
[20]:https://www.piskelapp.com/
|
||||||
|
[21]:https://github.com/piskelapp/piskel/blob/master/LICENSE
|
||||||
|
[22]:https://raw.githubusercontent.com/MUGeeksandGadgets/MarkMyWords/master/tools/unpiskel.py
|
||||||
|
[23]:https://www.python.org/
|
||||||
|
[24]:https://www.sacredchao.net/~piman/angrydd/
|
||||||
|
[25]:https://renpy.org/
|
||||||
|
[26]:https://www.Pygame.org/
|
||||||
|
[27]:http://Pygame.org/docs/tut/PygameIntro.html
|
||||||
|
[28]:https://anthony-tuininga.github.io/cx_Freeze/
|
||||||
|
[29]:https://Pygame.org/wiki/Pygame2exe
|
||||||
|
[30]:http://www.py2exe.org/
|
||||||
|
[31]:https://itch.io/jam/open-jam-1/entries
|
||||||
|
[32]:https://github.com/MUGeeksandGadgets/MarkMyWords/blob/3e1e8aed12ebe13acccf0d87b06d4f3bd124b9db/README.md#credits
|
||||||
|
[33]:https://twitter.com/mwcz
|
||||||
|
[34]:https://twitter.com/caramelcode
|
||||||
|
[35]:https://opensource.com/
|
@ -1,21 +1,22 @@
|
|||||||
糖尿病患者们是怎样使用开源造出自己的医疗设备的
|
糖尿病患者们是怎样使用开源造出自己的医疗设备的
|
||||||
======
|
======
|
||||||
|
> Red Hat 的 2018 女性开源社区奖获得者 Dana Lewis 的故事。
|
||||||
|
|
||||||
![](https://opensource.com/sites/default/files/styles/image-full-size/public/lead-images/health_heartbeat.png?itok=P-GXea-p)
|
![](https://opensource.com/sites/default/files/styles/image-full-size/public/lead-images/health_heartbeat.png?itok=P-GXea-p)
|
||||||
|
|
||||||
Dana Lewis 被评选为开源社区 2018 年度最佳女性!下面是开源怎样改善了她的健康的故事。
|
Dana Lewis 被评选为[开源社区 2018 年度最佳女性][1]!下面是开源怎样改善了她的健康的故事。
|
||||||
|
|
||||||
Dana 患有1型糖尿病,但当时市面上流通的药品和医疗设备都对她无效。
|
Dana 患有 I 型糖尿病,但当时市面上流通的药品和医疗设备都对她无效。她用来管理血糖的动态血糖监测(CGM)报警器的声音太小了,根本叫不醒熟睡的她,产品这样的设计无法保证她每天睡眠时间的生命安全。
|
||||||
她用来管理血糖的动态血糖监测报警器的声音太小了,根本叫不醒熟睡的她,产品这样的设计无法保证她每天睡眠时间的生命安全。
|
|
||||||
|
|
||||||
“我和生产厂家见了一面商议提出意见,厂家的回复是‘我们产品的音量已经足够大了,很少有人叫不醒’,我被告知‘这不是普遍问题,我们正在改进,请期待我们的新产品。’听到这些时我真的很挫败,但我从没想象过我能做出什么改变,毕竟那是通过了 FDA 标准的医疗设备,不是我们能随意改变的。”
|
“我和生产厂家见了一面商议提出意见,厂家的回复是‘我们产品的音量已经足够大了,很少有人叫不醒’,我被告知‘这不是普遍问题,我们正在改进,请期待我们的新产品。’听到这些时我真的很挫败,但我从没想象过我能做出什么改变,毕竟那是通过了 FDA 标准的医疗设备,不是我们能随意改变的。”
|
||||||
|
|
||||||
面临着这些阻碍,Dana 想着如果她能把自己的数据从设备里导出,就可以设置手机闹铃来叫醒自己。在2013年末,她看到的一条推特解决了她的疑问。那条推特的作者是一位糖尿病患儿的家长,他把动态血糖监测仪进行了逆向工程,这样就可以导出孩子的血糖数据进行远程监控了。
|
面临着这些阻碍,Dana 想着如果她能把自己的数据从设备里导出,就可以设置手机闹铃来叫醒自己。在 2013 年末,她看到的一条推特解决了她的疑问。那条推特的作者是一位糖尿病患儿的家长,他把动态血糖监测仪进行了逆向工程,这样就可以导出孩子的血糖数据进行远程监控了。
|
||||||
|
|
||||||
她意识到如果对方愿意把过程分享给她,她也可以用那些代码做一个自己的响亮的血糖监测仪了。
|
她意识到如果对方愿意把过程分享给她,她也可以用那些代码做一个自己的响亮的血糖监测仪了。
|
||||||
|
|
||||||
“我并不知道向别人要源代码是件稀松平常的事,那是我第一次接触开源。”
|
“我并不知道向别人要源代码是件稀松平常的事,那是我第一次接触开源。”
|
||||||
|
|
||||||
那个系统演化成一个响亮闹钟的代码,她也可以把代码在网页上分享给别人。和她的丈夫 Scott Leibrand 一起,她反复向闹铃添加属性,最终合成了一个算法,这个算法不仅能监测实时血糖水平,还能主动预测未来血糖波动。
|
那个系统演化成一个响亮闹钟的代码,她也可以把代码在网页上分享给别人。和她的丈夫 Scott Leibrand 一起,她逐步向闹铃添加属性,最终形成了一个算法,这个算法不仅能监测实时血糖水平,还能主动预测未来血糖波动。
|
||||||
|
|
||||||
随着 Dana 与开源糖尿病患者社区的接触越来越深,她认识了 Ben West,他花了很多年才研究出与 Dana 使用的胰岛素泵沟通数据的方法,与血糖监测仪不同,胰岛素泵不是简单的报告血糖,它是个单独的设备,要按人体需要持续推注胰岛素,比血糖监测仪要复杂得多。
|
随着 Dana 与开源糖尿病患者社区的接触越来越深,她认识了 Ben West,他花了很多年才研究出与 Dana 使用的胰岛素泵沟通数据的方法,与血糖监测仪不同,胰岛素泵不是简单的报告血糖,它是个单独的设备,要按人体需要持续推注胰岛素,比血糖监测仪要复杂得多。
|
||||||
|
|
||||||
@ -27,15 +28,15 @@ Dana 患有1型糖尿病,但当时市面上流通的药品和医疗设备都
|
|||||||
|
|
||||||
“正因为我们使用的是开源软件,在做出这个系统之后我们就把成果开源化了,这样可以造福更多的人。”开源人工胰腺系统 (OpenAPS) 由此诞生。
|
“正因为我们使用的是开源软件,在做出这个系统之后我们就把成果开源化了,这样可以造福更多的人。”开源人工胰腺系统 (OpenAPS) 由此诞生。
|
||||||
|
|
||||||
OpenAPS 社区已经拥有超过600名用户,大家都提供了各种各样的自制“闭路”系统代码。OpenAPS 贡献者们以 #WeAreNotWaiting 话题团结一致,以表达患者群体不该干等着医疗保健工厂制造出真正有效便捷产品的理念。
|
OpenAPS 社区已经拥有超过 600 名用户,大家都提供了各种各样的自制“闭路”系统代码。OpenAPS 贡献者们聚集到了 #WeAreNotWaiting 话题之下,以表达患者群体不该干等着医疗保健工厂制造出真正有效便捷产品的理念。
|
||||||
|
|
||||||
“你可以选择等待未来的商业解决方案,这无可厚非,选择等待是你的自由。等待可以是一种选择,但不能是无法改变的现状。对我来说,开源在医疗保健方面做出的这个举动让等待变成了一种选择。你可以选择不自行解决,你可以选择等待商业解决方案,但如果你不想等了,你无需再等。现在你有很多选择,开源社区的人们已经解决了很多问题。”
|
“你可以选择等待未来的商业解决方案,这无可厚非,选择等待是你的自由。等待可以是一种选择,但不能是无法改变的现状。对我来说,开源在医疗保健方面做出的这个举动让等待变成了一种选择。你可以选择不自行解决,你可以选择等待商业解决方案,但如果你不想等了,你无需再等。现在你有很多选择,开源社区的人们已经解决了很多问题。”
|
||||||
|
|
||||||
OpenAPS 社区由糖尿病患者,患者家属,还有想要合理利用这项技术的人们。在社区的帮助下,Dana 学会了很多种贡献开源项目的方式。她发现许多从 Facebook 或 [Gitter][2] 上过来的非技术贡献者也对 OpenAPS 做出了很大贡献。
|
OpenAPS 社区由糖尿病患者、患者家属,还有想要合理利用这项技术的人们。在社区的帮助下,Dana 学会了很多种贡献开源项目的方式。她发现许多从 Facebook 或 [Gitter][2] 上过来的非技术贡献者也对 OpenAPS 做出了很大贡献。
|
||||||
|
|
||||||
“贡献有很多方式,我们要认识到各种方式的贡献都是平等的。它们一般涉及不同的兴趣领域和技能组合,只有把这些综合起来,才能做成社区的项目。”
|
“贡献有很多方式,我们要认识到各种方式的贡献都是平等的。它们一般涉及不同的兴趣领域和技能组合,只有把这些综合起来,才能做成社区的项目。”
|
||||||
|
|
||||||
她亲身经历过,所以知道自己的贡献不被社区的其他成员认可是怎样难过的感受。对于人们习惯把女性的贡献打折的这一现象,她也不回避。在她的 [2014 年博客][3] 和 [反思][4] 文章中她初次写道在入围开源年度最佳人物时所遭受到的区别待遇,这些待遇让她意识到身为女性的不同。
|
她亲身经历过,所以知道自己的贡献不被社区的其他成员认可是怎样难过的感受。对于人们习惯把女性的贡献打折的这一现象,她也不回避。在她的 [2014 年博客][3] 和 [反思][4] 文章中她初次写到在入围开源年度最佳人物时所遭受到的区别待遇,这些待遇让她意识到身为女性的不同。
|
||||||
|
|
||||||
在她最初的博客中,她写道了自己和丈夫 Scott 同为开源社区成员,遭受到的区别待遇。他们都注意到,Dana 总是被提出一些细枝末节的要求,但 Scott 就不会。而 Scott 总被问道一些技术性问题,即使他向他们推荐 Dana,人们也更倾向于问身为男性的 Scott。大家都或多或少经历过这些行为,Dana 的博文在社区里引起了广泛的讨论。
|
在她最初的博客中,她写道了自己和丈夫 Scott 同为开源社区成员,遭受到的区别待遇。他们都注意到,Dana 总是被提出一些细枝末节的要求,但 Scott 就不会。而 Scott 总被问道一些技术性问题,即使他向他们推荐 Dana,人们也更倾向于问身为男性的 Scott。大家都或多或少经历过这些行为,Dana 的博文在社区里引起了广泛的讨论。
|
||||||
|
|
||||||
@ -45,13 +46,13 @@ OpenAPS 社区由糖尿病患者,患者家属,还有想要合理利用这项
|
|||||||
|
|
||||||
“我想如果我就放弃努力了,可能开源世界里糖尿病患者们的现状会有很大不同。我知道别人不幸的遭遇,他们在开源社区中感受不到认同感和自身价值,最终离开了开源。我希望我们可以继续这种讨论,大家都能意识到如果我们不故意打击贡献者,我们可以变得更加温暖,成员们也能感受到认同感,大家的付出也能得到相应的认可。
|
“我想如果我就放弃努力了,可能开源世界里糖尿病患者们的现状会有很大不同。我知道别人不幸的遭遇,他们在开源社区中感受不到认同感和自身价值,最终离开了开源。我希望我们可以继续这种讨论,大家都能意识到如果我们不故意打击贡献者,我们可以变得更加温暖,成员们也能感受到认同感,大家的付出也能得到相应的认可。
|
||||||
|
|
||||||
OpenAPS 社区的交流和分享给我们提供了一个很好的例子,它说明非技术性的贡献者对于整个社区的成功都是至关重要的。Dana 在现实社会中的关系和交流经历对她为开源社区做出的宣传有着很大的贡献。她为社区在 [DIYPS blog][5] 上写了很多篇文章,她还在 [TEDx Talk][6] 做过一场演讲, 在 [开源大会 (OSCON)][7] 上也演讲过很多次,诸如此类的还有很多。
|
OpenAPS 社区的交流和分享给我们提供了一个很好的例子,它说明非技术性的贡献者对于整个社区的成功都是至关重要的。Dana 在现实社会中的关系和交流经历对她为开源社区做出的宣传有着很大的贡献。她为社区在 [DIYPS 博客][5] 上写了很多篇文章,她还在 [TEDx Talk][6] 做过一场演讲, 在 [开源大会 (OSCON)][7] 上也演讲过很多次,诸如此类的还有很多。
|
||||||
|
|
||||||
“不是每个项目都像 OpenAPS 一样,对患者有那么大的影响,甚至成为患者中间的主流项目。糖尿病社区在项目的沟通中真的做了很多贡献,引来了很多糖尿病患者,也让需要帮助的人们知道了我们的存在。”
|
“不是每个项目都像 OpenAPS 一样,对患者有那么大的影响,甚至成为患者中间的主流项目。糖尿病社区在项目的沟通中真的做了很多贡献,引来了很多糖尿病患者,也让需要帮助的人们知道了我们的存在。”
|
||||||
|
|
||||||
Dana 现在的目标是帮助其他疾病的患者社区创建项目。她尤其想要把社区成员们学到的工具和技术和其他的患者社区分享,特别是那些想要把项目进一步提升,进行深入研究,或者想和公司合作的社区。
|
Dana 现在的目标是帮助其他疾病的患者社区创建项目。她尤其想要把社区成员们学到的工具和技术和其他的患者社区分享,特别是那些想要把项目进一步提升,进行深入研究,或者想和公司合作的社区。
|
||||||
|
|
||||||
“听说很多参与项目的患者都听过这样的话,‘你应该申请个专利;你应该拿它开个公司;你应该成立个非营利组织。’但这些都是大事,它们太耗时间了,不仅占据你的工作时间,甚至强行改变你的专业领域。我这样的人并不想做那样的事,我们更倾向于把精力放在壮大其他项目上,以此帮助更多的人。”
|
“我听说很多参与项目的患者都听过这样的话,‘你应该申请个专利;你应该拿它开个公司;你应该成立个非营利组织。’但这些都是大事,它们太耗时间了,不仅占据你的工作时间,甚至强行改变你的专业领域。我这样的人并不想做那样的事,我们更倾向于把精力放在壮大其他项目上,以此帮助更多的人。”
|
||||||
|
|
||||||
在此之后,她开始寻找其他不那么占用时间的任务,比如给小孩们写一本书。Dana 在 2017 年进行了这项挑战,她写了本书给侄子侄女,讲解他们婶婶的糖尿病设备是怎样工作的。在她侄女问她“胳膊上的东西是什么”(那是她的血糖监测仪)时,她意识到她不知道怎么和一个小孩子解释糖尿病患者是什么,所以写了[《卡罗琳的机器人亲戚》][8]这本书。
|
在此之后,她开始寻找其他不那么占用时间的任务,比如给小孩们写一本书。Dana 在 2017 年进行了这项挑战,她写了本书给侄子侄女,讲解他们婶婶的糖尿病设备是怎样工作的。在她侄女问她“胳膊上的东西是什么”(那是她的血糖监测仪)时,她意识到她不知道怎么和一个小孩子解释糖尿病患者是什么,所以写了[《卡罗琳的机器人亲戚》][8]这本书。
|
||||||
|
|
||||||
@ -70,7 +71,7 @@ via: https://opensource.com/article/18/5/dana-lewis-women-open-source-community-
|
|||||||
作者:[Taylor Greene][a]
|
作者:[Taylor Greene][a]
|
||||||
选题:[lujun9972](https://github.com/lujun9972)
|
选题:[lujun9972](https://github.com/lujun9972)
|
||||||
译者:[Valoniakim](https://github.com/Valoniakim)
|
译者:[Valoniakim](https://github.com/Valoniakim)
|
||||||
校对:[校对者ID](https://github.com/校对者ID)
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
77
published/201812/20180623 The IBM 029 Card Punch.md
Normal file
77
published/201812/20180623 The IBM 029 Card Punch.md
Normal file
@ -0,0 +1,77 @@
|
|||||||
|
IBM 029 型打孔机
|
||||||
|
======
|
||||||
|
|
||||||
|
我知道这很学院派,可一行超过 80 个字符的代码还是让我抓狂。我也在网上见过不少人认为即使在现代的视网膜屏幕下也应当采用行长度为 80 个字符的标准,可他们都不理解我对破坏这一标准的怒火,就算多 1 个字符也不行。
|
||||||
|
|
||||||
|
在这一标准的黄金时期,一行代码的长度几乎不会超过 80 个字符的限制。在那时,这一限制是物理的,没有第 81 列用于存放第 81 个字符。每一个试图把函数名起的又长又臭的程序员都会在短暂的愉悦后迎来更多的麻烦,而这仅仅是因为没有足够的空间放下整个函数的声明。
|
||||||
|
|
||||||
|
这一黄金时期也是<ruby>打孔卡<rt>punch card</rt></ruby>编程时期。在 20 世纪 60 年代,IBM 打孔卡设立了标准,这个标准就是打孔卡的宽度为 80 列。这个 80 列标准在后来的电传打字机和哑终端时期得以延续,并逐渐成为操作系统中隐藏的细节。时至今日,即使我们用上了更大、更好的屏幕,偏向于使用更长的标识符而不是类似 `iswcntrl()` 这样令人难以猜测的函数名,可当你打开新的终端模拟器窗口时,默认的宽度依然是 80 个字符。
|
||||||
|
|
||||||
|
从 Quora 上的很多问题中可以发现,很多人并不能想象如何使用打孔卡给计算机编程。我承认,在很长的一段时间里我也不能理解打孔卡编程是如何工作的,因为这让我想到就像劳工一样不停的给这些打孔卡打孔。当然,这是一个误解,程序员不需要亲自给打孔卡打孔,就像是火车调度员不用亲自扳道岔。程序员们有<ruby>打孔机<rt>card punch machines</rt></ruby>(也被称为<ruby>键控打孔机<rt>key punches</rt></ruby>),这让他们可以使用打字机式的键盘给打孔卡打孔。这样的设备在 19 世纪 90 年代时就已经不是什么新技术了。
|
||||||
|
|
||||||
|
那时,最为广泛使用的打孔机之一便是 IBM 029 型打孔机。就算在今天,它也许是最棒的打孔机。
|
||||||
|
|
||||||
|
![][1]
|
||||||
|
|
||||||
|
IBM 029 型打孔机在 1964 年作为 IBM 的 System/360 大型电脑的配件发售的。System/360 是计算系统与外设所组成的一个系列,在 20 世纪 60 年代晚期,它几乎垄断了整个大型计算机市场。就像其它 System/360 外设一样,029 型打孔机也是个大块头。那时,计算机和家具的界限还很模糊,但 029 型打孔机可不是那种会占领你的整张桌子的机器。它改进自 026 型打孔机,增加了新的字符支持,如括号,总体上也更加安静。与前辈 026 型所展出 20 世纪 40 年代的圆形按钮与工业化的样貌相比,029 型的按键方正扁平、功能按键还有酷炫的蓝色高亮提示。它的另一个重要买点是它能够在<ruby>数字区<rt>numeric field</rt></ruby>左侧自动的填充 0 ,这证明了 JavaScript 程序员不是第一批懒得自己做<ruby>左填充<rt>left-padding</rt></ruby>的程序员。(LCTT 译注:这项功能需要额外的 4 张 <ruby>[标准模块系统卡](https://en.wikipedia.org/wiki/IBM_Standard_Modular_System)<rt>SMS card</rt></ruby>才能使用。例如设置数字区域长度为 6 列时,操作员只需要输入 73 ,打孔机会自动填充起始位置上的 4 个 0 ,故最终输出 000073。[更多信息](https://en.wikipedia.org/wiki/Keypunch#IBM_029_Card_Punch))
|
||||||
|
|
||||||
|
等等!你说的是 IBM 在 1964 年发布了全新的打孔机?你知道那张在贝尔实验室拍摄的 Unix 之父正在使用电传打字机的照片吗?那是哪一年的来着?1970?打孔机不是应该在 20 世纪 60 年代中期到晚期时就过时了吗?是的,你也许会奇怪,为什么直到 1984 年,IBM 的产品目录中还会出现 029 型打孔机的身影 [^1]。事实上,直到 20 世纪 70 年代,大多数程序员仍然在使用打孔卡编程。其实二战期间就已经有人在用电传打字机了,可那时并没能普及。客观的讲,电传打字机几乎和打孔卡一样古老。也许和你想象的恰恰相反,并不是电传打字机本身限制了它的普及,而是计算时间。人们拒绝使用电传打字机的原因是,它是可交互的,它和计算机使用<ruby>“在线”的传输方式<rt>"online" mode of communication</rt></ruby>。在以 Unix 为代表的分时操作系统被发明前,你和电脑的交互会被任何人的使用而打断,而这一点延迟通常意味着几千美元的损失。所以程序员们普遍选择离线地使用打孔机编程,再将打孔卡放入大型计算机中,作为<ruby>批任务<rt>batch job</rt></ruby>执行。在那时,还没有即廉价又可靠的存储设备,可打孔卡的廉价优势已经足够让它成为那时最流行的数据存储方式了。那时的程序是书架上一摞打孔卡而不是硬盘里的一堆文件。
|
||||||
|
|
||||||
|
那么实际使用 IBM 029 型打孔机是个什么样子呢?这很难向没有实际看过打孔卡的人解释。一张打孔卡通常有 12 行 80 列。打孔卡下面是从 1 到 9 的<ruby>数字行<rt>digit rows</rt></ruby>,打孔卡上的每一列都有这些行所对应的数字。最上面的三行是<ruby>空间行<rt>"zone" rows</rt></ruby>,通常由两行空白行和一行 0 行组成。第 12 行是打孔卡最顶部的行,接下来是 11 行,随后是从数字 0 到 9 所在的行。这个有点让人感到困惑的顺序的原因是打孔卡的上边缘被称为<ruby>12 边<rt>12 edge</rt></ruby>、下边缘被称为 <ruby>9 边<rt>9 edge</rt></ruby>。那时,为了让打孔卡便于整理,常常会剪去打孔卡的一个角。
|
||||||
|
|
||||||
|
![][2]
|
||||||
|
|
||||||
|
(LCTT 译注:可参考[EBCDIC 编码](https://zh.wikipedia.org/wiki/EBCDIC))
|
||||||
|
|
||||||
|
在打孔卡发明之初,孔洞的形状是圆形的,但是 IBM 最终意识到如果使用窄长方形作为孔洞,一张卡就可以放下更多的列了。每一列中孔洞的不同组合就可以表达不同的字符。像 029 型这样的拥有人性化设计的打孔机除了完成本质的打孔任务外,还会在打孔卡最上方打印出每一列所对应的字符。输入是数字就在对应的数字行上打孔。输入的是字母或符号就用一个在空间列的孔和一或俩个在数字列的孔的组合表示,例如字母 A 就用一个在第 12 空间行的空和一个数字 1 所在行的孔表示。这是一种顺序编码,在第一台打孔机被发明后,也叫 Hollerith 编码。这种编码只能表示相对较小的一套字符集,小写字母就没有包含在这套字符集中。如今一些聪明的工程师可能想知道为什么打卡不干脆使用二进制编码 —— 毕竟,有 12 行,你可以编码超过 4000 个字符。 使用 Hollerith 编码是因为它确保在单个列中出现不超过三个孔。这保留了卡的结构强度。二进制编码会带来太多的孔,会因为孔洞过于密集而断裂。
|
||||||
|
|
||||||
|
打孔卡也有不同。在 20 世纪 60 年代,80 列虽然是标准,但表达的方式不一定相同。基础打孔卡是无标注的,但用于 COBOL 编程的打孔卡会把最后的 8 列保留,供标识数保存使用。这一标识数可以在打孔卡被打乱 (例如一叠打孔卡掉在地上了) 后用于自动排序。此外,第 7 列被用于表示本张打孔卡上的是否与上一张打孔卡一起构成一条语句。也就是说当你真的对 80 字符的限制感到绝望的时候,还可以用两张卡甚至更多的卡拼接成一条长语句。用于 FORTRAN 编程的打孔卡和 COBOL 打孔卡类似,但是定义的列不同。大学里使用的打孔卡通常会由其计算机中心加上水印,其它的设计则会在如 [1976 年美国独立 200 周年][3] 的特殊场合才会加入。
|
||||||
|
|
||||||
|
最终,这些打孔卡都要被计算机读取和计算。IBM 出售的 System/360 大型计算机的外设 IBM 2540 可以以每分钟 1000 张打孔卡的速度读取这些卡片[^2] 。IBM 2540 使用电刷扫过每张打孔卡,电刷通过孔洞就可以接触到卡片后面的金属板完成一次读取。一旦读取完毕,System/360 大型计算机就会把每张打孔卡上的数据使用一种定长的 8 位编码保存,这种编码是<ruby>扩增二进式十进交换码<rt>Extended Binary Coded Decimal Interchange Code</rt></ruby>,简写为 EBCDIC 编码。它是一种二进制编码,可以追溯自早期打孔卡所使用的 BCDIDC 编码 —— 其 6 位编码使用低 4 位表示数字行,高 2 位表示空间行。程序员们在打孔卡上编写完程序后,会把卡片们交给计算机操作员,操作员们会把这些卡片放入 IBM 2540 ,再把打印结果交给程序员。那时的程序员大多都没有见过计算机长什么样。
|
||||||
|
|
||||||
|
程序员们真正能见到的是很多打孔机。029 型打孔机虽然不是计算机,但这并不意味着它不是一台复杂的机器。看看这个由<ruby>密歇根大学<rt>University of Michigan</rt></ruby>计算机中心在 1967 年制作的[教学视频][4],你就能更好的理解使用一台 029 型打孔机是什么情形了。我会尽可能在这里总结这段视频,但如果你不去亲自看看的话,你会错过许多惊奇和感叹。
|
||||||
|
|
||||||
|
029 型打孔机的结构围绕着一个打孔卡穿过机器的 U 形轨道开始。使用打孔机时,右手边也就是 U 形轨道的右侧顶部是<ruby>进卡卡槽<rt>hopper</rt></ruby>,使用前通常在里面放入一叠未使用的打孔卡。虽然 029 型打孔机主要使用 80 列打孔卡,但在需要的情况下也可以使用更小号的打孔卡。在打孔机的使用过程中,打孔卡离开轨道右上端的进卡卡槽,顺着 U 形轨道移动并最终进入左上端的<ruby>出卡卡槽<rt>stacker</rt></ruby>。这一流程可以保证出卡卡槽中的打孔卡按打孔时的先后顺序排列。
|
||||||
|
|
||||||
|
029 型打孔机的开关在桌面下膝盖高度的位置。在开机后,连按两次 “<ruby>装入<rt>FEED</rt></ruby>” 键让机器自动将打孔卡从进卡卡槽中取出并移动到机器内。 U 形轨道的底部是打孔机的核心区域,它由三个部分组成:右侧是等待区,中间是打孔操作区,左侧是阅读区。连按两次 “装入” 键,机器就会把一张打孔卡装入打孔机的打孔操作区,另一张打孔卡进入等待区。在打孔操作区上方有一个列数指示器来显示当前打孔所在的列的位置。这时,每按下一个按键,机器就会在打孔卡对应的位置打孔并在卡片的顶部打印按键对应的字符,随后将打孔卡向左移动一列。如果一张卡片的 80 列全部被打上了数据,这张卡片会被打孔操作区自动释放并进入阅读区,同时,一张新的打孔卡会被装入打孔操作区。如果没有打完全部的 80 列,可以使用 “<ruby>释放<rt>REL</rt></ruby>” 键完成上面的操作。
|
||||||
|
|
||||||
|
在打孔卡上打印对应的字符这一设计让人很容易分辨出错误。但就像密歇根大学的视频中警告的那样,打孔卡上修正一个错误可不像擦掉一个打印的字符然后再写上一个新的那样容易,因为计算机只会根据卡片上的孔来读取信息。因为被打出的孔不能被<ruby>复原<rt>unpunched</rt></ruby>,所以并不能直接退回一列然后再打上一个新的字符。打出更多的孔也只能让这一列的组合变成一个无效字符。IBM 029 型打孔机上虽然有一个可以让打孔卡回退一列的退格按键,但这个按键被放置在机器上而非在键盘上。这样的设计也许是为了阻止这个按键的使用,因为实际上很少有用户需要这个功能。
|
||||||
|
|
||||||
|
实际上,只有废弃错误的打孔卡再在新的打孔卡上重新打孔这一种修正错误的方式。这就是阅读区的用武之处了。当你发现打孔卡上的第 68 列出错时,你需要在新的打孔卡上小心的给前 67 列重新打孔,然后给第 68 列打上正确的字母。另一种操作方式是把带有错误信息的打孔卡放在阅读区,同时在打孔操作区载入一张新的打孔卡,然后按下 “<ruby>重复<rt>DUP</rt></ruby>” 按键直到列数指示器显示 68 列。这时按下正确的字符来修正错误。阅读区和重复按键使得 029 型打孔机很容易复制打孔卡上的内容。当然,这一功能的使用可能有各种各样的原因,但改错是最常见的。
|
||||||
|
|
||||||
|
(LCTT 译注:有一种说法是“补丁”这个用于对已经发布的软件进行修复的术语来源于对打孔纸带或打孔卡上打错的孔贴上补丁的做法。可能对于长长的一卷打孔纸带来说,由于个别字母的错误而整个废弃成本过高,会采用“补丁”的方式;而对于这种单张式的打孔卡来说,重新打印一张正确的更为方便。)
|
||||||
|
|
||||||
|
“重复”按键允许 029 型打孔机的操作员手动调用重复的函数。但是 029 型打孔机还可以设置为自动重复。当用于记录数据而不是编程时,这项功能十分有效。举个例子,当用打孔卡来记录大学生的信息时,每张卡片上都需要输入学生的宿舍楼的名字,如果发现所输入信息的学生都在同一栋楼,就可以使用 029 型打孔机的自动重复功能来完成宿舍楼名称的填写。
|
||||||
|
|
||||||
|
像这样的自动化操作可以通过<ruby>程序鼓<rt>program drum</rt></ruby>编程到 029 型打孔机里面。程序鼓就安装在打孔操作区上方的 U 形轨道中间部分的右上角。通过在打孔卡上写下程序,然后把打孔卡装入程序鼓中,就完成了一次给 029 型打孔机的编程任务。用户可以通过这种方式对打孔卡的每一列都按需要定义不同的自动化操作。029 型打孔机允许指定某些列重复上一张打孔卡相同位置的字符,这就是它能更快的输入学生信息的理由。它还允许指定某些列只能输入数字或者字母,指定特定的列为空或者到某一列时就直接跳过一整张打孔卡。编程鼓使它在打孔特定列有特殊含义的固定模式卡片时很容易。密歇根大学制作的另一个[进阶教学视频][5]包括了给 029 型打孔机编程的过程,如果你已经掌握了它的基础操作,就快去看看吧。
|
||||||
|
|
||||||
|
这会儿,无论你是否看了密歇根大学制作的视频,都会感叹打孔机的操作之简便。虽然修正错误的过程很乏味,但除此之外,操作一台打孔机并不像想象的那样复杂。我甚至可以想象打孔卡之间的无缝切换让 COBOL 和 FORTRAN 程序员忘记了他们的程序是打在不同的打孔卡上而不是写在一个连续的文本文件内。另一方面,思考一下打孔机是如何影响编程语言的发展也是很有趣的,虽然它仅仅是一台输入设备。结构化编程最终会出现并鼓励程序员把整个代码块视为一个整体,但可以想象打孔卡程序员们强调每一行的作用且难以认同结构化编程的场景。同时你能够理解他们为什么不把代码块闭合所使用的括号放在单独的一行,只是因为这样会浪费打孔卡。
|
||||||
|
|
||||||
|
现在,虽然没有人再使用打孔卡编程了,每个程序员都该试试[这个][6],哪怕一次也好。或许你因此能够更好的理解 COBOL 和 FORTRAN 的历史,或许你就能体会到为什么每个人把 80 个字符作为长度限制的标注。
|
||||||
|
|
||||||
|
喜欢吗?这里每两周都会发表一篇这样的文章。请在推特上关注我们 [@TwoBitHistory][7] 或者订阅我们的 [RSS][8],这样你就能在第一时间收到新文章的通知。
|
||||||
|
|
||||||
|
[^1]: “IBM 29 Card Punch,” IBM Archives, accessed June 23, 2018, https://www-03.ibm.com/ibm/history/exhibits/vintage/vintage_4506VV4002.html.
|
||||||
|
[^2]: IBM, IBM 2540 Component Description and Operation Procedures (Rochester, MN: IBM Product Publications, 1965), September 06, 2009, accessed June 23, 2018, http://bitsavers.informatik.uni-stuttgart.de/pdf/ibm/25xx/A21-9033-1_2540_Card_Punch_Component_Description_1965.pdf.
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: https://twobithistory.org/2018/06/23/ibm-029-card-punch.html
|
||||||
|
|
||||||
|
作者:[Two-Bit History][a]
|
||||||
|
选题:[lujun9972][b]
|
||||||
|
译者:[wwhio](https://github.com/wwhio)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]: https://twobithistory.org
|
||||||
|
[b]: https://github.com/lujun9972
|
||||||
|
[1]: https://twobithistory.org/images/ibm029_front.jpg
|
||||||
|
[2]: https://twobithistory.org/images/card.png
|
||||||
|
[3]: http://www.jkmscott.net/data/Punched%20card%20013.jpg
|
||||||
|
[4]: https://www.youtube.com/watch?v=kaQmAybWn-w
|
||||||
|
[5]: https://www.youtube.com/watch?v=SWD1PwNxpoU
|
||||||
|
[6]: http://www.masswerk.at/keypunch/
|
||||||
|
[7]: https://twitter.com/TwoBitHistory
|
||||||
|
[8]: https://twobithistory.org/feed.xml
|
@ -1,51 +1,58 @@
|
|||||||
用户,组和其他 Linux 用户
|
用户、组及其它 Linux 特性
|
||||||
======
|
======
|
||||||
|
|
||||||
|
> Linux 和其他类 Unix 操作系统依赖于用户组,而不是逐个为用户分配权限和特权。一个组就是你想象的那样:一群在某种程度上相关的用户。
|
||||||
|
|
||||||
![](https://www.linux.com/sites/lcom/files/styles/rendered_file/public/flamingo-2458782_1920.jpg?itok=_gkzGGx5)
|
![](https://www.linux.com/sites/lcom/files/styles/rendered_file/public/flamingo-2458782_1920.jpg?itok=_gkzGGx5)
|
||||||
|
|
||||||
到这个阶段,[在看到如何操作目录或文件夹之后][1],但在让自己一头扎进文件之前,我们必须重新审视 _权限_, _users_ 和 _group_。幸运的是,[有一个网站上已经有了一个优秀而全面的教程,包括了权限][2],所以你应该去立刻阅读它。简而言之,你使用权限来确定谁可以对文件和目录执行操作,以及他们可以对每个文件和目录执行什么操作 -- 从中读取,写入,擦除等等。
|
到这个阶段,[在看到如何操作目录或文件夹之后][1],但在让自己一头扎进文件之前,我们必须重新审视 _权限_、_用户_ 和 _组_。幸运的是,[有一个网站上已经有了一个优秀而全面的教程,讲到了权限][2],所以你应该去立刻阅读它。简而言之,你使用权限来确定谁可以对文件和目录执行操作,以及他们可以对每个文件和目录执行什么操作 —— 从中读取、写入、移动、擦除等等。
|
||||||
|
|
||||||
要尝试本教程涵盖的所有内容,你需要在系统上创建新用户。让我们实践起来,为每一个需要借用你电脑的人创建一个用户,我们称之为 _guest 账户_。
|
要尝试本教程涵盖的所有内容,你需要在系统上创建新用户。让我们实践起来,为每一个需要借用你电脑的人创建一个用户,我们称之为 `guest` 账户。
|
||||||
|
|
||||||
**警告:** _例如,如果你错误地删除了自己的用户和目录,那么创建,特别是删除用户以及主目录会严重损坏系统。你可能不想在你日常的工作机中练习,那么请在另一台机器或者虚拟机上练习。无论你是否想要安全地练习,经常备份你的东西总是一个好主意。检查备份是否正常工作,为你自己以后避免很多咬牙切齿的事情。_
|
**警告:** 例如,如果你错误地删除了自己的用户和目录,那么创建用户,特别是删除用户以及主目录会严重损坏系统。你可能不想在你日常的工作机中练习,那么请在另一台机器或者虚拟机上练习。无论你是否想要安全地练习,经常备份你的东西总是一个好主意。检查备份是否正常工作,为你自己以后避免很多咬牙切齿的事情。
|
||||||
|
|
||||||
### 一个新用户
|
### 一个新用户
|
||||||
|
|
||||||
你可以使用 `useradd` 命令来创建一个新用户。使用超级用户或 root 权限运行 `useradd`,即使用 `sudo` 或 `su`,这具体取决于你的系统,你可以:
|
你可以使用 `useradd` 命令来创建一个新用户。使用超级用户或 root 权限运行 `useradd`,即使用 `sudo` 或 `su`,这具体取决于你的系统,你可以:
|
||||||
|
|
||||||
```
|
```
|
||||||
sudo useradd -m guest
|
sudo useradd -m guest
|
||||||
```
|
```
|
||||||
|
|
||||||
然后输入你的密码。或者也可以这样:
|
然后输入你的密码。或者也可以这样:
|
||||||
|
|
||||||
```
|
```
|
||||||
su -c "useradd -m guest"
|
su -c "useradd -m guest"
|
||||||
```
|
```
|
||||||
|
|
||||||
然后输入 root 或超级用户的密码。
|
然后输入 root 或超级用户的密码。
|
||||||
|
|
||||||
(_为了简洁起见,我们将从现在开始假设你使用 `sudo` 获得超级用户或 root 权限。_)
|
( _为了简洁起见,我们将从现在开始假设你使用 `sudo` 获得超级用户或 root 权限。_ )
|
||||||
|
|
||||||
通过使用 `-m` 参数,`useradd` 将为新用户创建一个主目录。你可以通过列出 _/home/guest_ 来查看其内容。
|
通过使用 `-m` 参数,`useradd` 将为新用户创建一个主目录。你可以通过列出 `/home/guest` 来查看其内容。
|
||||||
|
|
||||||
然后你可以使用以下命令来为新用户设置密码:
|
然后你可以使用以下命令来为新用户设置密码:
|
||||||
|
|
||||||
```
|
```
|
||||||
sudo passwd guest
|
sudo passwd guest
|
||||||
```
|
```
|
||||||
|
|
||||||
或者你也可以使用 `adduser`,这是一个交互式的命令,它会询问你一些问题,包括你要为用户分配的 shell(是的,不止一个),你希望其主目录在哪里,你希望他们属于哪些组(有关这点稍后会讲到)等等。在运行 `adduser` 结束时,你可以设置密码。注意,默认情况下,在许多发行版中都没有安装 `adduser`,但安装了 `useradd`。
|
或者你也可以使用 `adduser`,这是一个交互式的命令,它会询问你一些问题,包括你要为用户分配的 shell(是的,shell 有不止一种),你希望其主目录在哪里,你希望他们属于哪些组(有关这点稍后会讲到)等等。在运行 `adduser` 结束时,你可以设置密码。注意,默认情况下,在许多发行版中都没有安装 `adduser`,但安装了 `useradd`。
|
||||||
|
|
||||||
顺便说一下,你可以使用 `userdel` 来移除一个用户:
|
顺便说一下,你可以使用 `userdel` 来移除一个用户:
|
||||||
|
|
||||||
```
|
```
|
||||||
sudo userdel -r guest
|
sudo userdel -r guest
|
||||||
```
|
```
|
||||||
|
|
||||||
使用 `-r` 选项,`userdel` 不仅删除了 _guest_ 用户,还删除了他们的主目录和邮件中的条目(如果有的话)。
|
使用 `-r` 选项,`userdel` 不仅删除了 `guest` 用户,还删除了他们的主目录和邮件中的条目(如果有的话)。
|
||||||
|
|
||||||
### home 中的内容
|
### 主目录中的内容
|
||||||
|
|
||||||
谈到用户的主目录,它依赖于你所使用的发行版。你可能已经注意到,当你使用 `-m` 选项时,`useradd` 使用子目录填充用户的目录,包括音乐,文档和诸如此类的内容以及各种各样的隐藏文件。要查看 guest 主目录中的所有内容,运行 `sudo ls -la /home/guest`。
|
谈到用户的主目录,它依赖于你所使用的发行版。你可能已经注意到,当你使用 `-m` 选项时,`useradd` 使用子目录填充用户的目录,包括音乐、文档和诸如此类的内容以及各种各样的隐藏文件。要查看 `guest` 主目录中的所有内容,运行 `sudo ls -la /home/guest`。
|
||||||
|
|
||||||
|
进入新用户目录的内容通常是由 `/etc/skel` 架构目录确定的。有时它可能是一个不同的目录。要检查正在使用的目录,运行:
|
||||||
|
|
||||||
进入新用户目录的内容通常是由 _/etc/skel_ 架构目录确定的。有时它可能是一个不同的目录。要检查正在使用的目录,运行:
|
|
||||||
```
|
```
|
||||||
useradd -D
|
useradd -D
|
||||||
GROUP=100
|
GROUP=100
|
||||||
@ -57,31 +64,36 @@ SKEL=/etc/skel
|
|||||||
CREATE_MAIL_SPOOL=no
|
CREATE_MAIL_SPOOL=no
|
||||||
```
|
```
|
||||||
|
|
||||||
这给你一些额外的有趣信息,但你现在感兴趣的是 `SKEL=/etc/skel` 这一行,在这种情况下,按照惯例,它指向 _/etc/skel/_。
|
这会给你一些额外的有趣信息,但你现在感兴趣的是 `SKEL=/etc/skel` 这一行,在这种情况下,按照惯例,它指向 `/etc/skel/`。
|
||||||
|
|
||||||
|
由于 Linux 中的所有东西都是可定制的,因此你可以更改那些放入新创建的用户目录的内容。试试这样做:在 `/etc/skel/` 中创建一个新目录:
|
||||||
|
|
||||||
由于 Linux 中的所有东西都是可定制的,因此你可以更改那些放入新创建的用户目录的内容。试试这样做:在 _/etc/skel/_ 中创建一个新目录:
|
|
||||||
```
|
```
|
||||||
sudo mkdir /etc/skel/Documents
|
sudo mkdir /etc/skel/Documents
|
||||||
```
|
```
|
||||||
|
|
||||||
然后创建一个包含欢迎消息的文件,并将其复制过来:
|
然后创建一个包含欢迎消息的文件,并将其复制过来:
|
||||||
|
|
||||||
```
|
```
|
||||||
sudo cp welcome.txt /etc/skel/Documents
|
sudo cp welcome.txt /etc/skel/Documents
|
||||||
```
|
```
|
||||||
|
|
||||||
现在删除 guest 账户:
|
现在删除 `guest` 账户:
|
||||||
|
|
||||||
```
|
```
|
||||||
sudo userdel -r guest
|
sudo userdel -r guest
|
||||||
```
|
```
|
||||||
|
|
||||||
再次创建:
|
再次创建:
|
||||||
|
|
||||||
```
|
```
|
||||||
sudo useradd -m guest
|
sudo useradd -m guest
|
||||||
```
|
```
|
||||||
|
|
||||||
嘿 presto!(to 校正:这个 presto 是什么?)你的 _Documents/_ 目录和 _welcome.txt_ 文件神奇地出现在了 guest 的主目录中。
|
嘿!你的 `Documents/` 目录和 `welcome.txt` 文件神奇地出现在了 `guest` 的主目录中。
|
||||||
|
|
||||||
|
你还可以在创建用户时通过编辑 `/etc/default/useradd` 来修改其他内容。我的看起来像这样:
|
||||||
|
|
||||||
你还可以在创建用户时通过编辑 _/etc/default/useradd_ 来修改其他内容。我的看起来像这样:
|
|
||||||
```
|
```
|
||||||
GROUP=users
|
GROUP=users
|
||||||
HOME=/home
|
HOME=/home
|
||||||
@ -96,11 +108,12 @@ CREATE_MAIL_SPOOL=no
|
|||||||
|
|
||||||
### 群组心态
|
### 群组心态
|
||||||
|
|
||||||
Linux 和其他类 Unix 操作系统依赖于 _groups_,而不是逐个为用户分配权限和特权。一个组就是你想象的那样:一群在某种程度上相关的用户。在你的系统上可能有一组允许使用打印机的用户,他们属于 _lp_(即 "_line printer_")组。传统上 _wheel_ 组的成员是唯一可以通过使用 _su_ 成为超级用户或 root 的成员。_network_ 用户组可以启动或关闭网络。还有许多诸如此类的。
|
Linux 和其他类 Unix 操作系统依赖于用户组,而不是逐个为用户分配权限和特权。一个组就是你想象的那样:一群在某种程度上相关的用户。在你的系统上可能有一组允许使用打印机的用户,他们属于 `lp`(即 “_line printer_”)组。传统上 `wheel` 组的成员是唯一可以通过使用 `su` 成为超级用户或 root 的成员。`network` 用户组可以启动或关闭网络。还有许多诸如此类的。
|
||||||
|
|
||||||
不同的发行版有不同的组,具有相同或相似名称的组具有不同的权限,这也取决于你使用的发行版。因此,如果你在前一段中读到的内容与你系统中的内容不匹配,不要感到惊讶。
|
不同的发行版有不同的组,具有相同或相似名称的组具有不同的权限,这也取决于你使用的发行版。因此,如果你在前一段中读到的内容与你系统中的内容不匹配,不要感到惊讶。
|
||||||
|
|
||||||
不管怎样g,要查看系统中有哪些组,你可以使用:
|
不管怎样,要查看系统中有哪些组,你可以使用:
|
||||||
|
|
||||||
```
|
```
|
||||||
getent group
|
getent group
|
||||||
```
|
```
|
||||||
@ -108,18 +121,20 @@ getent group
|
|||||||
`getent` 命令列出了某些系统数据库的内容。
|
`getent` 命令列出了某些系统数据库的内容。
|
||||||
|
|
||||||
要查找当前用户所属的组,尝试:
|
要查找当前用户所属的组,尝试:
|
||||||
|
|
||||||
```
|
```
|
||||||
groups
|
groups
|
||||||
```
|
```
|
||||||
|
|
||||||
当你使用 `useradd` 创建新用户时,除非你另行指定,否则用户讲只属于一个组:他们自己。一个 _guest_ 用户属于 _guest_ 组。组使用户有权管理自己的东西,仅此而已。
|
当你使用 `useradd` 创建新用户时,除非你另行指定,否则用户将只属于一个组:他们自己。`guest` 用户属于 `guest` 组。组使用户有权管理自己的东西,仅此而已。
|
||||||
|
|
||||||
你可以使用 `groupadd` 命令创建新组,然后添加用户:
|
你可以使用 `groupadd` 命令创建新组,然后添加用户:
|
||||||
|
|
||||||
```
|
```
|
||||||
sudo groupadd photos
|
sudo groupadd photos
|
||||||
```
|
```
|
||||||
|
|
||||||
例如,这将创建 _photos_ 组。下一次,我们将使用它来构建一个共享目录,该组的所有成员都可以读取和写入,我们将更多地了解权限和特权。敬请关注!
|
例如,这将创建 `photos` 组。下一次,我们将使用它来构建一个共享目录,该组的所有成员都可以读取和写入,我们将更多地了解权限和特权。敬请关注!
|
||||||
|
|
||||||
|
|
||||||
--------------------------------------------------------------------------------
|
--------------------------------------------------------------------------------
|
||||||
@ -129,11 +144,11 @@ via: https://www.linux.com/learn/intro-to-linux/2018/7/users-groups-and-other-li
|
|||||||
作者:[Paul Brown][a]
|
作者:[Paul Brown][a]
|
||||||
选题:[lujun9972](https://github.com/lujun9972)
|
选题:[lujun9972](https://github.com/lujun9972)
|
||||||
译者:[MjSeven](https://github.com/MjSeven)
|
译者:[MjSeven](https://github.com/MjSeven)
|
||||||
校对:[校对者ID](https://github.com/校对者ID)
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
[a]:https://www.linux.com/users/bro66
|
[a]:https://www.linux.com/users/bro66
|
||||||
[1]:https://www.linux.com/blog/learn/2018/5/manipulating-directories-linux
|
[1]:https://linux.cn/article-10066-1.html
|
||||||
[2]:https://www.linux.com/learn/understanding-linux-file-permissions
|
[2]:https://www.linux.com/learn/understanding-linux-file-permissions
|
||||||
[3]:https://training.linuxfoundation.org/linux-courses/system-administration-training/introduction-to-linux
|
[3]:https://training.linuxfoundation.org/linux-courses/system-administration-training/introduction-to-linux
|
@ -0,0 +1,114 @@
|
|||||||
|
用户、组及其它 Linux 特性(二)
|
||||||
|
======
|
||||||
|
> 我们继续创建和管理用户和组的 Linux 教程之旅。
|
||||||
|
|
||||||
|
![](https://www.linux.com/sites/lcom/files/styles/rendered_file/public/ducks-94911_1920.jpg?itok=7_ZPiph7)
|
||||||
|
|
||||||
|
在正在进行的 Linux 之旅中,我们了解了[如何操作文件夹或目录][1],现在我们继续讨论 _权限_、_用户_ 和 _组_,这对于确定谁可以操作哪些文件和目录是必要的。[上次][2],我们展示了如何创建新用户,现在我们将重新起航:
|
||||||
|
|
||||||
|
你可以使用 `groupadd` 命令创建新组,然后随意添加用户。例如,使用:
|
||||||
|
|
||||||
|
```
|
||||||
|
sudo groupadd photos
|
||||||
|
```
|
||||||
|
|
||||||
|
这将会创建 `photos` 组。
|
||||||
|
|
||||||
|
你需要在根目录下[创建一个目录][1]:
|
||||||
|
|
||||||
|
```
|
||||||
|
sudo mkdir /photos
|
||||||
|
```
|
||||||
|
|
||||||
|
如果你运行 `ls -l /`,结果中会有如下这一行:
|
||||||
|
|
||||||
|
```
|
||||||
|
drwxr-xr-x 1 root root 0 jun 26 21:14 photos
|
||||||
|
```
|
||||||
|
|
||||||
|
输出中的第一个 `root` 是所属的用户,第二个 `root` 是所属的组。
|
||||||
|
|
||||||
|
要将 `/photos` 目录的所有权转移到 `photos` 组,使用:
|
||||||
|
|
||||||
|
```
|
||||||
|
chgrp photos /photos
|
||||||
|
```
|
||||||
|
|
||||||
|
`chgrp` 命令通常采用两个参数,第一个参数是将要获得文件或目录所有权的组,第二个参数是希望交给组的文件或目录。
|
||||||
|
|
||||||
|
接着,运行 `ls -l /`,你会发现刚才那一行变了:
|
||||||
|
|
||||||
|
```
|
||||||
|
drwxr-xr-x 1 root photos 0 jun 26 21:14 photos
|
||||||
|
```
|
||||||
|
|
||||||
|
你已成功将新目录的所有权转移到了 `photos` 组。
|
||||||
|
|
||||||
|
然后,将你自己的用户和 `guest` 用户添加到 `photos` 组:
|
||||||
|
|
||||||
|
```
|
||||||
|
sudo usermod <你的用户名> -a -G photos
|
||||||
|
sudo usermod guest -a -G photos
|
||||||
|
```
|
||||||
|
|
||||||
|
你可能必须注销并重新登录才能看到更改,但是当你这样做时,运行 `groups` 会将 `photos` 显示为你所属的组之一。
|
||||||
|
|
||||||
|
关于上面提到的 `usermod` 命令,需要指明几点。第一:注意要使用 `-G` 选项而不是 `-g` 选项。`-g` 选项更改你的主要组,如果你意外地使用它,它可能会锁定你的一些东西。另一方面,`-G` 将你添加到列出的组中,并没有干扰主要组。如果要将用户添加到多个组中,在 `-G` 之后逐个列出它们,用逗号分隔,不要有空格:
|
||||||
|
|
||||||
|
```
|
||||||
|
sudo usermod <your username> -a -G photos,pizza,spaceforce
|
||||||
|
```
|
||||||
|
|
||||||
|
第二点:小心点不要忘记 `-a` 参数。`-a` 参数代表追加,将你传递给 `-G` 的组列表附加到你已经属于的组。这意味着,如果你不包含 `-a`,那么你之前所属的组列表将被覆盖,再次将你拒之门外。
|
||||||
|
|
||||||
|
这些都不是灾难性问题,但这意味着你必须手动将用户添加回你所属的所有组,这可能是个麻烦,特别是如果你失去了对 `sudo` 和 `wheel` 组的访问权限。
|
||||||
|
|
||||||
|
### 权限
|
||||||
|
|
||||||
|
在将图像复制到 `/photos` 目录之前,还要做一件事情。注意,当你执行上面的 `ls -l /` 时,该文件夹的权限将以 `drwxr-xr-x` 形式返回。
|
||||||
|
|
||||||
|
如果你阅读[我在本文开头推荐的文章][3],你将知道第一个 `d` 表示文件系统中的条目是一个目录,接着你有三组三个字符(`rwx`、`r-x`、`r-x`),它们表示目录的所属用户(`rwx`)的权限,然后是所属组(`r-x`)的权限,最后是其他用户(`r-x`)的权限。这意味着到目前为止唯一具有写权限的人,即能够在 `/photos` 目录中复制或创建文件的唯一人员是 `root` 用户。
|
||||||
|
|
||||||
|
但是[我提到的那篇文章也告诉你如何更改目录或文件的权限][3]:
|
||||||
|
|
||||||
|
```
|
||||||
|
sudo chmod g+w /photos
|
||||||
|
```
|
||||||
|
|
||||||
|
运行 `ls -l /`,你会看到 `/photos` 权限变为了 `drwxrwxr-x`。这就是你希望的:组成员现在可以对目录进行写操作了。
|
||||||
|
|
||||||
|
现在你可以尝试将图像或任何其他文件复制到目录中,它应该没有问题:
|
||||||
|
|
||||||
|
```
|
||||||
|
cp image.jpg /photos
|
||||||
|
```
|
||||||
|
|
||||||
|
`guest` 用户也可以从目录中读取和写入。他们也可以读取和写入,甚至移动或删除共享目录中其他用户创建的文件。
|
||||||
|
|
||||||
|
### 总结
|
||||||
|
|
||||||
|
Linux 中的权限和特权系统已经磨练了几十年,它继承自昔日的旧 Unix 系统。就其本身而言,它工作的非常好,而且经过了深思熟虑。熟悉它对于任何 Linux 系统管理员都是必不可少的。事实上,除非你理解它,否则你根本就无法做很多事情。但是,这并不难。
|
||||||
|
|
||||||
|
下一次,我们将深入研究文件,并以一个创新的方式查看创建,操作和销毁文件的不同方法。最后一个总是很有趣。
|
||||||
|
|
||||||
|
回头见!
|
||||||
|
|
||||||
|
通过 Linux 基金会和 edX 的免费[“Linux 简介”][4]课程了解有关 Linux 的更多信息。
|
||||||
|
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: https://www.linux.com/blog/learn/intro-to-linux/2018/7/users-groups-and-other-linux-beasts-part-2
|
||||||
|
|
||||||
|
作者:[Paul Brown][a]
|
||||||
|
选题:[lujun9972](https://github.com/lujun9972)
|
||||||
|
译者:[MjSeven](https://github.com/MjSeven)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]:https://www.linux.com/users/bro66
|
||||||
|
[1]:https://linux.cn/article-10066-1.html
|
||||||
|
[2]:https://linux.cn/article-10370-1.html
|
||||||
|
[3]:https://www.linux.com/learn/understanding-linux-file-permissions
|
||||||
|
[4]:https://training.linuxfoundation.org/linux-courses/system-administration-training/introduction-to-linux
|
@ -0,0 +1,142 @@
|
|||||||
|
[#]: collector: (lujun9972)
|
||||||
|
[#]: translator: (geekpi)
|
||||||
|
[#]: reviewer: (wxy)
|
||||||
|
[#]: publisher: (wxy)
|
||||||
|
[#]: url: (https://linux.cn/article-10375-1.html)
|
||||||
|
[#]: subject: (11 Uses for a Raspberry Pi Around the Office)
|
||||||
|
[#]: via: (https://blog.dxmtechsupport.com.au/11-uses-for-a-raspberry-pi-around-the-office/)
|
||||||
|
[#]: author: (James Mawson https://blog.dxmtechsupport.com.au/author/james-mawson/)
|
||||||
|
|
||||||
|
树莓派在办公室的 11 种用法
|
||||||
|
======
|
||||||
|
|
||||||
|
我知道你在想什么:树莓派只能用在修修补补、原型设计和个人爱好中。它实际不能用在业务中。
|
||||||
|
|
||||||
|
毫无疑问,这台电脑的处理能力相对较低、易损坏的 SD 卡、缺乏电池备份以及支持的 DIY 性质,这意味着它不会是一个能在任何时候执行最关键的操作的[专业的、已安装好、配置好的商业服务器][1]的可行替代品。
|
||||||
|
|
||||||
|
但是它电路板便宜、功耗很小、小到几乎适合任何地方、无限灵活 —— 这实际上是处理办公室一些基本任务的好方法。
|
||||||
|
|
||||||
|
而且,更好的是,已经有一些人完成了这些项目并很乐意分享他们是如何做到的。
|
||||||
|
|
||||||
|
### DNS 服务器
|
||||||
|
|
||||||
|
每次在浏览器中输入网站地址或者点击链接时,都需要将域名转换为数字 IP 地址,然后才能显示内容。
|
||||||
|
|
||||||
|
通常这意味着向互联网上某处 DNS 服务器发出请求 —— 但你可以通过本地处理来加快浏览速度。
|
||||||
|
|
||||||
|
你还可以分配自己的子域,以便本地访问办公室中的计算机。
|
||||||
|
|
||||||
|
[这里了解它是如何工作的。][2]
|
||||||
|
|
||||||
|
### 厕所占用标志
|
||||||
|
|
||||||
|
在厕所排过队吗?
|
||||||
|
|
||||||
|
这对于那些等待的人来说很烦人,花在处理它上面的时间会耗费你在办公室的工作效率。
|
||||||
|
|
||||||
|
我想你希望在办公室里也悬挂飞机上那个厕所有人的标志。
|
||||||
|
|
||||||
|
[Occu-pi][3] 是一个非常简单的解决方案,使用磁性开关和树莓派来判断螺栓何时关闭,并在 Slack 频道中更新“厕所在使用中” —— 这意味着整个办公室的人都可以看一眼电脑或者移动设备知道是否有空闲的隔间。
|
||||||
|
|
||||||
|
### 针对黑客的蜜罐陷阱
|
||||||
|
|
||||||
|
黑客破坏了网络的第一个线索是一些事情变得糟糕,这应该会吓到大多数企业主。
|
||||||
|
|
||||||
|
这就是可以用到蜜罐的地方:一台没有任何服务的计算机位于你的网络,将特定端口打开,伪装成黑客喜欢的目标。
|
||||||
|
|
||||||
|
安全研究人员经常在网络外部部署蜜罐,以收集攻击者正在做的事情的数据。
|
||||||
|
|
||||||
|
但对于普通的小型企业来说,这些作为一种绊脚石部署在内部更有用。因为普通用户没有真正的理由想要连接到蜜罐,所以任何发生的登录尝试都是正在进行捣乱的非常好的指示。
|
||||||
|
|
||||||
|
这可以提供对外部人员入侵的预警,并且也可以提供对值得信赖的内部人员的预警。
|
||||||
|
|
||||||
|
在较大的客户端/服务器网络中,将它作为虚拟机运行可能更为实用。但是在无线路由器上运行的点对点的小型办公室/家庭办公网络中,[HoneyPi][4] 之类的东西是一个很小的防盗报警器。
|
||||||
|
|
||||||
|
### 打印服务器
|
||||||
|
|
||||||
|
联网打印机更方便。
|
||||||
|
|
||||||
|
但更换所有打印机可能会很昂贵 —— 特别是如果你对现有的打印机感到满意的话。
|
||||||
|
|
||||||
|
[将树莓派设置为打印服务器][5]可能会更有意义。
|
||||||
|
|
||||||
|
### 网络附加存储(NAS)
|
||||||
|
|
||||||
|
将硬盘变为 NAS 是树莓派最早的实际应用之一,并且它仍然是最好的之一。
|
||||||
|
|
||||||
|
[这是如何使用树莓派创建 NAS。][6]
|
||||||
|
|
||||||
|
### 工单服务器
|
||||||
|
|
||||||
|
想要在预算不足的情况下在服务台中支持工单?
|
||||||
|
|
||||||
|
有一个名为 osTicket 的完全开源的工单程序,它可以安装在你的树莓派上,它甚至还有[随时可用的 SD 卡镜像][7]。
|
||||||
|
|
||||||
|
### 数字标牌
|
||||||
|
|
||||||
|
无论是用于活动、广告、菜单还是其他任何东西,许多企业都需要一种显示数字标牌的方式 —— 而树莓派的廉价和省电使其成为一个非常有吸引力的选择。
|
||||||
|
|
||||||
|
[这有很多可供选择的选项。] [8]
|
||||||
|
|
||||||
|
### 目录和信息亭
|
||||||
|
|
||||||
|
[FullPageOS][9] 是一个基于 Raspbian 的 Linux 发行版,它直接引导到 Chromium 的全屏版本 —— 这非常适合导购、图书馆目录等。
|
||||||
|
|
||||||
|
### 基本的内联网 Web 服务器
|
||||||
|
|
||||||
|
对于托管一个面向公众的网站,你最好有一个托管帐户。树莓派不适合面对真正的网络流量。
|
||||||
|
|
||||||
|
但对于小型办公室,它可以托管内部业务维基或基本的公司内网。它还可以用作沙箱环境,用于试验代码和服务器配置。
|
||||||
|
|
||||||
|
[这里是如何在树莓派上运行 Apache、MySQL 和 PHP。][10]
|
||||||
|
|
||||||
|
### 渗透测试器
|
||||||
|
|
||||||
|
Kali Linux 是专为探测网络安全漏洞而构建的操作系统。通过将其安装在树莓派上,你就拥有了一个超便携式穿透测试器,其中包含 600 多种工具。
|
||||||
|
|
||||||
|
[你可以在这里找到树莓派镜像的种子链接。][11]
|
||||||
|
|
||||||
|
绝对要小心只在你自己的网络或你有权对它安全审计的网络中使用它 —— 使用此方法来破解其他网络是严重的犯罪行为。
|
||||||
|
|
||||||
|
### VPN 服务器
|
||||||
|
|
||||||
|
当你外出时,依靠的是公共无线互联网,你无法控制还有谁在网络中、谁在窥探你的所有流量。这就是为什么通过 VPN 连接加密所有内容可以让人放心。
|
||||||
|
|
||||||
|
你可以订阅任意数量的商业 VPN 服务,并且你可以在云中安装自己的服务,但是在办公室运行一个 VPN,这样你也可以从任何地方访问本地网络。
|
||||||
|
|
||||||
|
对于轻度使用 —— 比如偶尔的商务旅行 —— 树莓派是一种强大的,节约能源的设置 VPN 服务器的方式。(首先要检查一下你的路由器是不是不支持这个功能,许多路由器是支持的。)
|
||||||
|
|
||||||
|
[这是如何在树莓派上安装 OpenVPN。][12]
|
||||||
|
|
||||||
|
### 无线咖啡机
|
||||||
|
|
||||||
|
啊,美味:好喝的饮料是神赐之物,也是公司内工作效率的支柱。
|
||||||
|
|
||||||
|
那么,为什么不[将办公室的咖啡机变成可以精确控制温度和无线连接的智能咖啡机呢?][13]
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: https://blog.dxmtechsupport.com.au/11-uses-for-a-raspberry-pi-around-the-office/
|
||||||
|
|
||||||
|
作者:[James Mawson][a]
|
||||||
|
选题:[lujun9972][b]
|
||||||
|
译者:[geekpi](https://github.com/geekpi)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]: https://blog.dxmtechsupport.com.au/author/james-mawson/
|
||||||
|
[b]: https://github.com/lujun9972
|
||||||
|
[1]: https://dxmtechsupport.com.au/server-configuration
|
||||||
|
[2]: https://www.1and1.com/digitalguide/server/configuration/how-to-make-your-raspberry-pi-into-a-dns-server/
|
||||||
|
[3]: https://blog.usejournal.com/occu-pi-the-bathroom-of-the-future-ed69b84e21d5
|
||||||
|
[4]: https://trustfoundry.net/honeypi-easy-honeypot-raspberry-pi/
|
||||||
|
[5]: https://opensource.com/article/18/3/print-server-raspberry-pi
|
||||||
|
[6]: https://howtoraspberrypi.com/create-a-nas-with-your-raspberry-pi-and-samba/
|
||||||
|
[7]: https://everyday-tech.com/a-raspberry-pi-ticketing-system-image-with-osticket/
|
||||||
|
[8]: https://blog.capterra.com/7-free-and-open-source-digital-signage-software-options-for-your-next-event/
|
||||||
|
[9]: https://github.com/guysoft/FullPageOS
|
||||||
|
[10]: https://maker.pro/raspberry-pi/projects/raspberry-pi-web-server
|
||||||
|
[11]: https://www.offensive-security.com/kali-linux-arm-images/
|
||||||
|
[12]: https://medium.freecodecamp.org/running-your-own-openvpn-server-on-a-raspberry-pi-8b78043ccdea
|
||||||
|
[13]: https://www.techradar.com/au/how-to/how-to-build-your-own-smart-coffee-machine
|
@ -0,0 +1,153 @@
|
|||||||
|
如何在 VirtualBox 上安装并使用 FreeDOS?
|
||||||
|
======
|
||||||
|
|
||||||
|
> 这份指南将带你如何一步一步在 Linux 平台下利用 VirtualBox 安装 FreeDOS。
|
||||||
|
|
||||||
|
### Linux 下借助 VirtualBox 安装 FreeDOS
|
||||||
|
|
||||||
|
- [How to Install FreeDOS in Linux using Virtual Box](https://www.youtube.com/p1MegqzFAqA)
|
||||||
|
|
||||||
|
2017 年的 11 月份,我[采访了 Jim Hall][1] 关于 [FreeDOS 项目][2] 背后的历史故事。今天,我将告诉你如何安装并使用 FreeDOS。需要注意到是:我将在 [Solus][4](一种针对家庭用户的 Linux 桌面发行版)下使用 5.2.14 版本的 [VirtualBox][3] 来完成这些操作。
|
||||||
|
|
||||||
|
> 注意:在本教程我将使用 Solus 作为主机系统因为它很容易设置。另一个你需要注意的事情是 Solus 的软件中心有两个版本的 VirtualBox:`virtualbox` 和 `virtualbox-current`。Solus 会让你选择是使用 linux-lts 内核还是 linux-current 内核。最终区别就是,`virtualbox` 适用于 linux-lts 而 `virtualbx-current` 适用于 linux-current。
|
||||||
|
|
||||||
|
#### 第一步 – 创建新的虚拟机
|
||||||
|
|
||||||
|
![][5]
|
||||||
|
|
||||||
|
当你打开 VirtualBox,点击 “New” 按钮来新建一个虚拟机。你可以自定义这台虚拟机的名字,我将它命名为 “FreeDOS”。你也可以在标注栏内指明你正在安装的 FreeDOS 的版本。你还需要选择你将要安装的操作系统的类型和版本。选择 “Other” 下的 “DOS”。
|
||||||
|
|
||||||
|
#### 第二步 – 设置内存大小
|
||||||
|
|
||||||
|
![][6]
|
||||||
|
|
||||||
|
下一个对话框会问你要给 FreeDOS 主机分配多少可用的内存空间。默认分配 32 MB。不必更改它。在 DOS 系统盛行的年代,32 MB 大小的内存对于一台搭载 FreeDOS 的机器已经很足够了。如果你有需要,你可以通过对你针对 FreeDOS 新建的虚拟机右键并选择 “Setting -> Symtem” 来增加内存。
|
||||||
|
|
||||||
|
![][7]
|
||||||
|
|
||||||
|
#### 第三步 – 创建虚拟硬盘
|
||||||
|
|
||||||
|
![][8]
|
||||||
|
|
||||||
|
下一步,你会被要求创建一个虚拟硬盘用来存储 FreeDOS 和它的文件。如果你还没有创建,只需要点击 “Create”。
|
||||||
|
|
||||||
|
下一个对话框会问你想用什么磁盘文件类型。默认的类型 (VirtualBox Disk Image) 效果就挺好。点击 “Next”。
|
||||||
|
|
||||||
|
下一个你遇到的问题是你想虚拟硬盘以何种方式创建。你是否希望虚拟硬盘占据的空间刚开始很小然后会随着你创建文件和安装软件逐渐增加直至达到你设置的上限?那么选择动态分配。如果你更喜欢虚拟硬盘 (VHD) 按照既定大小直接创建,选择固定大小即可。如果你不打算使用整个 VHD 或者你的硬盘空余空间不是太足够,那么动态分配是个很不错的分配方式。(需要注意的是,动态分配的虚拟硬盘占据的空间会随着你增加文件而增加,但不会因为你删除文件而变小) 我个人更喜欢动态分配,但你可以根据实际需要来选择最合适你的分配类型然后点击 “Next”。
|
||||||
|
|
||||||
|
![][9]
|
||||||
|
|
||||||
|
现在,你可以选择虚拟磁盘的大小和位置。500 MB 已经很足够了。需要注意的是很多你之后用到的程序都是基于文本的,这意味着它们占据的空间非常小。在你做好这些调整后,点击 “Create”。
|
||||||
|
|
||||||
|
#### 第四步 – 关联 .iso 文件
|
||||||
|
|
||||||
|
在我们继续之前,你需要[下载][10] FreeDOS 的 .iso 文件。你需要选择 CDROM 格式的 “standard” 安装程序。
|
||||||
|
|
||||||
|
![][11]
|
||||||
|
|
||||||
|
当文件下载完毕后,返回到 VirtualBox。选中你的虚拟机并打开设置。你可以通过对虚拟机右键并选中 “Setting” 或者选中虚拟机并点击 “Setting” 按钮。
|
||||||
|
|
||||||
|
接下来,点击 “Storage” 选项卡。在 “Storage Devices” 下面,选中 CD 图标。(它应该会在图标旁边显示 “Empty”。) 在右边的 “Attribute” 面板,点中 CD 图标然后在对应路径选中你刚下载的 .iso 文件。
|
||||||
|
|
||||||
|
> 提示:通常,在你通过 VirtualBox 安装完一个操作系统后你就可以删除对应的 .iso 文件了。但这并不适合 FreeDOS 。如果你想通过 FreeDOS 的包管理器来安装应用程序,你需要这个 .iso 文件。我通常会让这个 .iso 文件连接到虚拟机以便我安装一些程序。如果你也这么做了,你必须要确认下你让 FreeDOS 虚拟机每次启动的时候是从硬盘启动因为虚拟机的默认设置是从已关联的 .iso 文件启动。如果你忘了关联 .iso 文件,也不用担心。你可以通过选择 FreeDOS 虚拟机窗口上方的 “Devices” 来关联。然后就会发现 .iso 文件列在 “Optical Drives”。
|
||||||
|
|
||||||
|
#### 第五步 – 安装 FreeDOS
|
||||||
|
|
||||||
|
![][12]
|
||||||
|
|
||||||
|
既然我们已经完成了所有的准备工作,让我们来开始安装 FreeDOS 吧。
|
||||||
|
|
||||||
|
首先,你需要知道关于最新版本的 VirtualBox 的一个 bug。当我们创建好虚拟硬盘然后选中 “Install to harddisk” 后,如果你开启虚拟机你会发现在 FreeDOS 的欢迎界面出现过后就是不断滚动无群无尽的机器代码。我最近就遇到过这个问题而且不管是 Linux 还是 Windows 平台的 VirtualBox 都会碰到这个问题。(我知道解决办法。)
|
||||||
|
|
||||||
|
为了避开这个问题,你需要做一个简单的修改。当你看到 FreeDOS 的欢迎界面的时候,按下 Tab 键。(确认 “Install to harddrive” 已经选中。)在 “fdboot.img” 之后输入 `raw` 然后按下回车键。接下来就会启动 FreeDOS 的安装程序。
|
||||||
|
|
||||||
|
![][13]
|
||||||
|
|
||||||
|
安装程序会首先处理你的虚拟磁盘的格式化。当格式化完成后,安装程序会重启。当 FreeDOS 的欢迎界面再次出现的时候,你必须重新输入 `raw` 就像你之前输入的内容那样。
|
||||||
|
|
||||||
|
要确保在安装过程中你遇到的所有问题你选的都是 “Yes”。但也要注意有一个很重要的问题:“What FreeDOS packages do you want to install?” 的答案并不是 “Yes” 或者 “No”。答案有两个选择分别是 “Base packages” 和 “Full installation”。“Base packages” 针对的是想体验类似原始的 MS-DOS 环境的人群。“Full installation” 则包括了一系列工具和实用的程序来提升 DOS。
|
||||||
|
|
||||||
|
在整个安装过程的最后,你可以选择重启或者继续停留在 DOS。选择“reboot”。
|
||||||
|
|
||||||
|
#### 第六步 – 设置网络
|
||||||
|
|
||||||
|
不同于原始的 DOS,FreeDOS 可以访问互联网。你可以安装新的软件包或者更新你已经安装的软件包。要想使用网络,你还需要在 FreeDOS 安装些应用程序。
|
||||||
|
|
||||||
|
![][14]
|
||||||
|
|
||||||
|
首先,启动进入你新创建的 FreeDOS 虚拟机。在 FreeDOS 的选择界面,选中 “Boot from System harddrive”。
|
||||||
|
|
||||||
|
![][15]
|
||||||
|
|
||||||
|
现在,你可以通过输入 `fdimples` 来访问 FreeDOS 的软件包管理工具。你也可以借助方向键来浏览软件包管理器,然后用空格键选择类别或者软件包。在 “Networking” 类别中,你需要选中 `fdnet`。FreeDOS project 推荐也安装 `mtcp` 和 `wget`。多次点击 Tab 键直到选中 “OK” 然后在按下回车键。安装完成后,输入 `reboot` 并按下回车键确认执行。系统重启后,引导你的系统驱动。如果网络安装成功的话,你会在终端看到一些关于你的网络信息的新消息。
|
||||||
|
|
||||||
|
![][16]
|
||||||
|
|
||||||
|
注意:
|
||||||
|
|
||||||
|
有时候 VirtualBox 的默认设置并没有生效。如果遇到这种情况,先关闭你的 FreeDOS 虚拟机窗口。在 VirtualBox 主界面右键你的虚拟机并选中 “Setting”。VirtualBox 默认的网络设置是 “NAT”。将它改为 “Bridged Adapter” 后再尝试安装 FreeDOS 的软件包。现在就应该能正常运作了。
|
||||||
|
|
||||||
|
#### 第七步 – FreeDOS 的基本使用
|
||||||
|
|
||||||
|
##### 常见命令
|
||||||
|
|
||||||
|
既然你已经成功安装了 FreeDOS,让我们来看些基础命令。如果你已经在 Windows 平台使用过命令提示符,那么你会发现有很多命令都是相似的。
|
||||||
|
|
||||||
|
* `DIR`– 显示当前目录的内容
|
||||||
|
* `CD` – 改变当前所在的目录
|
||||||
|
* `COPY OLD.TXT NEW.TXT`– 复制文件
|
||||||
|
* `TYPE TEST.TXT` – 显示文件内容
|
||||||
|
* `DEL TEST.TXT` – 删除文件
|
||||||
|
* `XCOPY DIR NEWDIR` – 复制目录及目录下的所有内容
|
||||||
|
* `EDIT TEST.TXT`– 编辑一个文件
|
||||||
|
* `MKDIR NEWDIR` – 创建一个新目录
|
||||||
|
* `CLS` – 清除屏幕
|
||||||
|
|
||||||
|
你可以借助互联网或者 Jim Hall 所创建的 [方便的速查表][17] 来找到更多基本的 DOS 命令。
|
||||||
|
|
||||||
|
##### 运行一个程序
|
||||||
|
|
||||||
|
在 FreeDOS 上运行程序相当简单。需要注意的是当你借助 `fdimples` 软件包管理器来安装一个应用程序的时候,要确保你指定了待安装程序的 .EXE 文件的路径。这个路径会在应用程序的详细信息中显示。要想运行程序,通常你还需要进入到程序所在文件夹并输入该程序的名字。
|
||||||
|
|
||||||
|
例如,FreeDOS 中你可以安装一个叫 `FED` 的编辑器。安装完成后,你还需要做的就是进入 `C:\FED` 这个文件夹下并输入 `FED`。
|
||||||
|
|
||||||
|
对于位于 `\bin` 这个文件夹的程序,比如 Pico。这些程序可以在任意文件夹中被调用。
|
||||||
|
|
||||||
|
对于游戏通常会有一个或者两个 .EXE 程序,你玩游戏之前不得不先运行它们。这些设置文件通常能够修复你遇到的声音,视频,或者控制问题。
|
||||||
|
|
||||||
|
如果你遇到一些本教程中没指出的问题,别忘记访问 [FreeDOS 主站][2] 来寻求解决办法。他们有一个 wiki 和一些其他的支持选项。
|
||||||
|
|
||||||
|
你使用过 FreeDOS 吗?你还想看关于 FreeDOS 哪些方面的教程?请在下面的评论区告诉我们。
|
||||||
|
|
||||||
|
如果你觉得本篇文章很有趣,请花一分钟的时间将它分享在你的社交媒体,Hacker News 或者 [Reddit][18]。
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: https://itsfoss.com/install-freedos/
|
||||||
|
|
||||||
|
作者:[John Paul][a]
|
||||||
|
选题:[lujun9972](https://github.com/lujun9972)
|
||||||
|
译者:[WangYueScream](https://github.com/WangYueScream)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]: https://itsfoss.com/author/john/
|
||||||
|
[1]:https://itsfoss.com/interview-freedos-jim-hall/
|
||||||
|
[2]:http://www.freedos.org/
|
||||||
|
[3]:https://www.virtualbox.org/
|
||||||
|
[4]:https://solus-project.com/home/
|
||||||
|
[5]:https://i2.wp.com/itsfoss.com/wp-content/uploads/2018/07/freedos-tutorial-1.jpg?w=787&ssl=1
|
||||||
|
[6]:https://i0.wp.com/itsfoss.com/wp-content/uploads/2018/07/freedos-tutorial-2.jpg?w=792&ssl=1
|
||||||
|
[7]:https://i2.wp.com/itsfoss.com/wp-content/uploads/2018/07/freedos-tutorial-3.jpg?w=797&ssl=1
|
||||||
|
[8]:https://i1.wp.com/itsfoss.com/wp-content/uploads/2018/07/freedos-tutorial-4.jpg?w=684&ssl=1
|
||||||
|
[9]:https://i1.wp.com/itsfoss.com/wp-content/uploads/2018/07/freedos-tutorial-6.jpg?w=705&ssl=1
|
||||||
|
[10]:http://www.freedos.org/download/
|
||||||
|
[11]:https://i0.wp.com/itsfoss.com/wp-content/uploads/2018/07/freedos-tutorial-7.jpg?w=800&ssl=1
|
||||||
|
[12]:https://i1.wp.com/itsfoss.com/wp-content/uploads/2018/07/freedos-tutorial-8.png?w=789&ssl=1
|
||||||
|
[13]:https://i1.wp.com/itsfoss.com/wp-content/uploads/2018/07/freedos-tutorial-9.png?w=748&ssl=1
|
||||||
|
[14]:https://i1.wp.com/itsfoss.com/wp-content/uploads/2018/07/freedos-tutorial-10.png?w=792&ssl=1
|
||||||
|
[15]:https://i2.wp.com/itsfoss.com/wp-content/uploads/2018/07/freedos-tutorial-11.png?w=739&ssl=1
|
||||||
|
[16]:https://i0.wp.com/itsfoss.com/wp-content/uploads/2018/07/freedos-tutorial-12.png?w=744&ssl=1
|
||||||
|
[17]:https://opensource.com/article/18/6/freedos-commands-cheat-sheet
|
||||||
|
[18]:http://reddit.com/r/linuxusersgroup
|
@ -1,24 +1,25 @@
|
|||||||
量子计算的开源框架 Cirq 介绍
|
量子计算的开源框架 Cirq 介绍
|
||||||
======
|
======
|
||||||
|
|
||||||
我们即将讨论的内容正如标题所示,本文通过使用 Cirq 的一个开源视角,尝试去了解我们已经在量子计算领域取得多大的成就,和该领域的发展方向,以加快科学和技术研究。
|
我们即将讨论的内容正如标题所示,本文通过使用 Cirq 的一个开源视角,尝试去了解我们已经在量子计算领域取得多大的成就,和该领域的发展方向,以加快科学和技术研究。
|
||||||
|
|
||||||
首先,我们将引领你进入量子计算的世界。在我们深入了解 Cirq 在未来的量子计算中扮演什么样的重要角色之前,我们将尽量向你解释其背后的基本概念。Cirq,你最近可能听说过,在这个领域中已经发生了重大新闻,在 Open Science 上的文章中,我们将去尝试找出答案。
|
首先,我们将引领你进入量子计算的世界。在我们深入了解 Cirq 在未来的量子计算中扮演什么样的重要角色之前,我们将尽量向你解释其背后的基本概念。你最近可能听说过,在这个领域中有件重大新闻,就是 Cirq。在这篇开放科学栏目的文章中,我们将去尝试找出答案。
|
||||||
|
|
||||||
<https://www.youtube.com/embed/WVv5OAR4Nik?enablejsapi=1&autoplay=0&cc_load_policy=0&iv_load_policy=1&loop=0&modestbranding=1&rel=0&showinfo=0&fs=1&playsinline=0&autohide=2&theme=dark&color=red&controls=2&>
|
- [How it Works - Quantum Computing](https://www.youtube.com/WVv5OAR4Nik)
|
||||||
|
|
||||||
在我们开始了解量子计算之前,必须先去了解“量子”这个术语,量子是已知的 [亚原子粒子][1] 中最小的物质。[量子][2] 这个词来自拉丁语 Quantus,意思是 “有多少”,在下面的短视频链接中有描述:
|
在我们开始了解量子计算之前,必须先去了解“量子”这个术语,量子是已知的 [亚原子粒子][1] 中最小的物质。<ruby>[量子][2]<rt>Quantum</rt></ruby> 这个词来自拉丁语 Quantus,意思是 “有多小”,在下面的短视频链接中有描述:
|
||||||
|
|
||||||
<https://www.youtube.com/embed/-pUOxVsxu3o?enablejsapi=1&autoplay=0&cc_load_policy=0&iv_load_policy=1&loop=0&modestbranding=1&rel=0&showinfo=0&fs=1&playsinline=0&autohide=2&theme=dark&color=red&controls=2&>
|
- [What is a quantum Why is it significant](https://www.youtube.com/-pUOxVsxu3o)
|
||||||
|
|
||||||
为了易于我们理解量子计算,我们将量子计算与<ruby>经典计算<rt>Classical Computing</rt></ruby>(也有译做传统计算)进行比较。经典计算是指设计用于工作的、正在使用的计算机,正如你现在用于阅读本文的设备,就是我们所谓的经典计算设备。
|
为了易于我们理解量子计算,我们将<ruby>量子计算<rt>Quantum Computing</rt></ruby>与<ruby>经典计算<rt>Classical Computing</rt></ruby>(LCTT 译注:也有译做“传统计算”)进行比较。经典计算是指今天的传统计算机如何设计工作的,正如你现在用于阅读本文的设备,就是我们所谓的经典计算设备。
|
||||||
|
|
||||||
### 经典计算
|
### 经典计算
|
||||||
|
|
||||||
经典计算是描述计算机如何工作的另一种方式。它们通过一个二进制系统工作,即信息使用 1 或 0 来存储。经典计算机不会理解除 1 或 0 之外的任何其它东西。
|
经典计算只是描述计算机如何工作的另一种方式。它们通过一个二进制系统工作,即信息使用 1 或 0 来存储。经典计算机不会理解除 1 或 0 之外的任何其它东西。
|
||||||
|
|
||||||
直白来说,在计算机内部一个晶体管只能是开(1)或关(0)。我们输入的任何信息都被转换为无数个 1 和 0,所以计算机只能理解和存储 1 和 0。所有的东西都只能用无数个 1 和 0 的组合来表示。
|
直白来说,在计算机内部一个晶体管只能是开(1)或关(0)。我们输入的任何信息都被转换为无数个 1 和 0,以便计算机能理解和存储。所有的东西都只能用无数个 1 和 0 的组合来表示。
|
||||||
|
|
||||||
<https://www.youtube.com/embed/Xpk67YzOn5w?enablejsapi=1&autoplay=0&cc_load_policy=0&iv_load_policy=1&loop=0&modestbranding=1&rel=0&showinfo=0&fs=1&playsinline=0&autohide=2&theme=dark&color=red&controls=2&>
|
- [Why Do Computers Use 1s and 0s Binary and Transistors Explained](https://www.youtube.com/Xpk67YzOn5w)
|
||||||
|
|
||||||
### 量子计算
|
### 量子计算
|
||||||
|
|
||||||
@ -26,39 +27,39 @@
|
|||||||
|
|
||||||
请注意,叠加和纠缠 [不是同一个现象][4]。
|
请注意,叠加和纠缠 [不是同一个现象][4]。
|
||||||
|
|
||||||
<https://www.youtube.com/embed/jiXuVIEg10Q?enablejsapi=1&autoplay=0&cc_load_policy=0&iv_load_policy=1&loop=0&modestbranding=1&rel=0&showinfo=0&fs=1&playsinline=0&autohide=2&theme=dark&color=red&controls=2&>
|
- [How Do Quantum Computers Work!](https://www.youtube.com/jiXuVIEg10Q)
|
||||||
|
|
||||||
![][5]
|
![][5]
|
||||||
|
|
||||||
就像在经典计算中,我们有<ruby>比特<rt>bit</rt></ruby>,在量子计算中,我们相应也有<ruby>量子比特<rt>qubits</rt></ruby>(或 Quantum bits)。想了解它们二者之间的巨大差异之处,请查看这个 [页面][6],从那里的图片中可以得到答案。
|
就像在经典计算中,我们有<ruby>比特<rt>bit</rt></ruby>,在量子计算中,我们相应也有<ruby>量子比特<rt>qubit</rt></ruby>(即 Quantum bit)。想了解它们二者之间的巨大差异之处,请查看这个 [页面][6],从那里的图片中可以得到答案。
|
||||||
|
|
||||||
量子计算机并不是来替代我们的经典计算机的。但是,有一些非常巨大的任务用我们的经典计算机是无法完成的,而那些正是量子计算机大显身手的好机会。下面链接的视频详细描述了上述情况,同时也描述了量子计算机的原理。
|
量子计算机并不是来替代我们的经典计算机的。但是,有一些非常巨大的任务用我们的经典计算机是无法完成的,而那些正是量子计算机大显身手的好机会。下面链接的视频详细描述了上述情况,同时也描述了量子计算机的原理。
|
||||||
|
|
||||||
<https://www.youtube.com/embed/JhHMJCUmq28?enablejsapi=1&autoplay=0&cc_load_policy=0&iv_load_policy=1&loop=0&modestbranding=1&rel=0&showinfo=0&fs=1&playsinline=0&autohide=2&theme=dark&color=red&controls=2&>
|
- [Quantum Computers Explained – Limits of Human Technology](https://www.youtube.com/JhHMJCUmq28)
|
||||||
|
|
||||||
下面的视频全面描述了量子计算领域到目前为止的最新进展:
|
下面的视频全面描述了量子计算领域到目前为止的最新进展:
|
||||||
|
|
||||||
<https://www.youtube.com/embed/CeuIop_j2bI?enablejsapi=1&autoplay=0&cc_load_policy=0&iv_load_policy=1&loop=0&modestbranding=1&rel=0&showinfo=0&fs=1&playsinline=0&autohide=2&theme=dark&color=red&controls=2&>
|
- [Quantum Computing 2018 Update](https://www.youtube.com/CeuIop_j2bI)
|
||||||
|
|
||||||
### 嘈杂中型量子
|
### 嘈杂中型量子
|
||||||
|
|
||||||
根据最新更新的(2018 年 7 月 31 日)研究论文,术语 “Noisy” 是指由于对量子比特未能完全控制所产生的不准确性。正是这种不准确性严重制约了量子设备短期内实现其目标。
|
根据最新更新的(2018 年 7 月 31 日)研究论文,术语 “<ruby>嘈杂<rt>Noisy</rt></ruby>” 是指由于对量子比特未能完全控制所产生的不准确性。正是这种不准确性在短期内严重制约了量子设备实现其目标。
|
||||||
|
|
||||||
“中型” 指的是在接下来的几年中,量子计算机将要实现的量子规模大小,届时,量子比特的数目将可能从 50 到几百个不等。50 个量子比特是一个重大的量程碑,因为它将超越现有的最强大的 [超级计算机][8] 的 [暴力][7] 模拟能力。更多信息请阅读 [这里的][9] 论文。
|
“中型” 指的是在接下来的几年中,量子计算机将要实现的量子规模大小,届时,量子比特的数目将可能从 50 到几百个不等。50 个量子比特是一个重大的量程碑,因为它将超越现有的最强大的 [超级计算机][8] 的 [暴力破解][7] 所能比拟的计算能力。更多信息请阅读 [这里的][9] 论文。
|
||||||
|
|
||||||
随着 Cirq 出现,许多事情将会发生变化。
|
随着 Cirq 出现,许多事情将会发生变化。
|
||||||
|
|
||||||
### Cirq 是什么?
|
### Cirq 是什么?
|
||||||
|
|
||||||
Cirq 是一个 python 框架,它用于创建、编辑和调用我们前面讨论的嘈杂中型量子(NISQ)。换句话说,Cirq 能够解决挑战,去改善精确度和降低量子计算中的噪声。
|
Cirq 是一个 Python 框架,它用于创建、编辑和调用我们前面讨论的嘈杂中型量子(NISQ)。换句话说,Cirq 能够解决挑战,去改善精确度和降低量子计算中的噪声。
|
||||||
|
|
||||||
Cirq 并不需要必须有一台真实的量子计算机。Cirq 能够使用一个类似模拟器的界面去执行量子电路模拟。
|
Cirq 并不需要必须有一台真实的量子计算机。Cirq 能够使用一个类似模拟器的界面去执行量子电路模拟。
|
||||||
|
|
||||||
Cirq 的前进步伐越来越快了,[Zapata][10] 是使用它的首批用户之一,Zapata 是由来自哈佛大学的一群专注于量子计算的科学家在去年成立的。
|
Cirq 的前进步伐越来越快了,[Zapata][10] 是使用它的首批用户之一,Zapata 是由来自哈佛大学的专注于量子计算的[一群科学家][11]在去年成立的。
|
||||||
|
|
||||||
### Linux 上使用 Cirq 入门
|
### Linux 上使用 Cirq 入门
|
||||||
|
|
||||||
开源的 [Cirq 库][12] 开发者建议将它安装在像 [virtualenv][14] 这样的一个 [虚拟 python 环境][13] 中。在 Linux 上的开发者安装指南可以在 [这里][15] 找到。
|
开源的 [Cirq 库][12] 开发者建议将它安装在像 [virtualenv][14] 这样的一个 [虚拟 Python 环境][13] 中。在 Linux 上的开发者安装指南可以在 [这里][15] 找到。
|
||||||
|
|
||||||
但我们在 Ubuntu 16.04 的系统上成功地安装和测试了 Python3 的 Cirq 库,安装步骤如下:
|
但我们在 Ubuntu 16.04 的系统上成功地安装和测试了 Python3 的 Cirq 库,安装步骤如下:
|
||||||
|
|
||||||
@ -66,43 +67,41 @@ Cirq 的前进步伐越来越快了,[Zapata][10] 是使用它的首批用户
|
|||||||
|
|
||||||
![Cirq Framework for Quantum Computing in Linux][16]
|
![Cirq Framework for Quantum Computing in Linux][16]
|
||||||
|
|
||||||
首先,我们需要 pip 或 pip3 去安装 Cirq。[Pip][17] 是推荐用于安装和管理 Python 包的工具。
|
首先,我们需要 `pip` 或 `pip3` 去安装 Cirq。[Pip][17] 是推荐用于安装和管理 Python 包的工具。
|
||||||
|
|
||||||
对于 Python 3.x 版本,Pip 能够用如下的命令来安装:
|
对于 Python 3.x 版本,Pip 能够用如下的命令来安装:
|
||||||
|
|
||||||
```
|
```
|
||||||
sudo apt-get install python3-pip
|
sudo apt-get install python3-pip
|
||||||
|
|
||||||
```
|
```
|
||||||
|
|
||||||
Python3 包能够通过如下的命令来安装:
|
Python3 包能够通过如下的命令来安装:
|
||||||
|
|
||||||
```
|
```
|
||||||
pip3 install <package-name>
|
pip3 install <package-name>
|
||||||
|
|
||||||
```
|
```
|
||||||
|
|
||||||
我们继续去使用 Pip3 为 Python3 安装 Cirq 库:
|
我们继续去使用 Pip3 为 Python3 安装 Cirq 库:
|
||||||
|
|
||||||
```
|
```
|
||||||
pip3 install cirq
|
pip3 install cirq
|
||||||
|
|
||||||
```
|
```
|
||||||
|
|
||||||
#### 启用 Plot 和 PDF 生成(可选)
|
#### 启用 Plot 和 PDF 生成(可选)
|
||||||
|
|
||||||
可选系统的依赖没有安装的,可以使用 pip 去安装它:
|
可选系统的依赖没有被 Pip 安装的,可以使用如下命令去安装它:
|
||||||
|
|
||||||
```
|
```
|
||||||
sudo apt-get install python3-tk texlive-latex-base latexmk
|
sudo apt-get install python3-tk texlive-latex-base latexmk
|
||||||
|
|
||||||
```
|
```
|
||||||
|
|
||||||
* python3-tk 是 Python 自有的启用了绘图功能的图形库
|
* python3-tk 是 Python 自有的启用了绘图功能的图形库
|
||||||
* texlive-latex-base 和 latexmk 启动了 PDF 输出功能。
|
* texlive-latex-base 和 latexmk 启动了 PDF 输出功能。
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
最后,我们使用如下的命令和代码成功测试了 Cirq:
|
最后,我们使用如下的命令和代码成功测试了 Cirq:
|
||||||
|
|
||||||
```
|
```
|
||||||
python3 -c 'import cirq; print(cirq.google.Foxtail)'
|
python3 -c 'import cirq; print(cirq.google.Foxtail)'
|
||||||
|
|
||||||
```
|
```
|
||||||
|
|
||||||
我们得到的输出如下图:
|
我们得到的输出如下图:
|
||||||
@ -113,27 +112,27 @@ python3 -c 'import cirq; print(cirq.google.Foxtail)'
|
|||||||
|
|
||||||
我们也配置了一个 Python IDE [PyCharm][19] 去测试同样的结果:
|
我们也配置了一个 Python IDE [PyCharm][19] 去测试同样的结果:
|
||||||
|
|
||||||
因为在我们的 Linux 系统上为 Python3 安装了 Cirq,我们在 IDE 中配置项目解释器路径:
|
因为在我们的 Linux 系统上为 Python3 安装了 Cirq,我们在 IDE 中配置项目解释器路径为:
|
||||||
|
|
||||||
```
|
```
|
||||||
/usr/bin/python3
|
/usr/bin/python3
|
||||||
|
|
||||||
```
|
```
|
||||||
|
|
||||||
![][20]
|
![][20]
|
||||||
|
|
||||||
在上面的输出中,你可能注意到我们刚设置的项目解释器路径与测试程序文件(test.py)的路径显示在一起。退出代码 0 表示程序已经成功退出,没有错误。
|
在上面的输出中,你可能注意到我们刚设置的项目解释器路径与测试程序文件(`test.py`)的路径显示在一起。退出代码 0 表示程序已经成功退出,没有错误。
|
||||||
|
|
||||||
因此,那是一个现成的 IDE 环境,你可以导入 Cirq 库去开始使用 Python 去编程和模拟量子电路。
|
因此,那是一个已经就绪的 IDE 环境,你可以导入 Cirq 库去开始使用 Python 去编程和模拟量子电路。
|
||||||
|
|
||||||
#### Cirq 使用入门
|
#### Cirq 使用入门
|
||||||
|
|
||||||
Criq 入门的一个好的开端就是它 GitHub 页面上的 [示例][21]。
|
Criq 入门的一个好的开端就是它 GitHub 页面上的 [示例][21]。
|
||||||
|
|
||||||
Cirq 的开发者在 GitHub 上已经放了学习 [教程][22]。如果你想认真地学习量子计算,他们推荐你去看一本非常好的书,它是[由 Nielsen 和 Chuang 写的名为 “量子计算和量子信息“][23]。
|
Cirq 的开发者在 GitHub 上已经放置了学习 [教程][22]。如果你想认真地学习量子计算,他们推荐你去看一本非常好的书,它是[由 Nielsen 和 Chuang 写的名为 《量子计算和量子信息》][23]。
|
||||||
|
|
||||||
#### OpenFermion-Cirq
|
#### OpenFermion-Cirq
|
||||||
|
|
||||||
[OpenFermion][24] 是一个开源库,它是为了在量子计算机上模拟获取和操纵代表的费米系统(包含量子化学)。根据 [粒子物理学][26] 理论,按照 [费米— 狄拉克统计][27],费米系统与 [费米子][25] 的产生相关。
|
[OpenFermion][24] 是一个开源库,它是为了在量子计算机上模拟获取和操纵代表的费米系统(包含量子化学)。根据 [粒子物理学][26] 理论,按照 [费米—狄拉克统计][27],费米系统与 [费米子][25] 的产生相关。
|
||||||
|
|
||||||
OpenFermion 被称为从事 [量子化学][29] 的化学家和研究人员的 [一个极好的实践工具][28]。量子化学主要专注于 [量子力学][30] 在物理模型和化学系统实验中的应用。量子化学也被称为 [分子量子力学][31]。
|
OpenFermion 被称为从事 [量子化学][29] 的化学家和研究人员的 [一个极好的实践工具][28]。量子化学主要专注于 [量子力学][30] 在物理模型和化学系统实验中的应用。量子化学也被称为 [分子量子力学][31]。
|
||||||
|
|
||||||
@ -141,7 +140,7 @@ Cirq 的出现使 OpenFermion 通过提供程序和工具去扩展功能成为
|
|||||||
|
|
||||||
#### Google Bristlecone
|
#### Google Bristlecone
|
||||||
|
|
||||||
2018 年 3 月 5 日,在洛杉矶举行的一年一度的 [美国物理学会会议][33] 上,Google 发布了 [Bristlecone][32],这是他们的最新的量子处理器。这个 [基于门的超导系统][34] 为 Google 提供了一个测试平台,用以研究 [量子比特技术][37] 的 [系统错误率][35] 和 [扩展性][36] ,以及在量子 [仿真][38]、[优化][39]、和 [机器学习][40] 方面的应用。
|
2018 年 3 月 5 日,在洛杉矶举行的一年一度的 [美国物理学会会议][33] 上,Google 发布了 [Bristlecone][32],这是他们的最新的量子处理器。这个 [基于门的超导系统][34] 为 Google 提供了一个测试平台,用以研究 [量子比特技术][37] 的 [系统错误率][35] 和 [扩展性][36] ,以及在量子 [仿真][38]、[优化][39] 和 [机器学习][40] 方面的应用。
|
||||||
|
|
||||||
Google 希望在不久的将来,能够制造出它的 [云可访问][41] 的 72 个量子比特的 Bristlecone 量子处理器。Bristlecone 将越来越有能力完成一个经典超级计算机无法在合理时间内完成的任务。
|
Google 希望在不久的将来,能够制造出它的 [云可访问][41] 的 72 个量子比特的 Bristlecone 量子处理器。Bristlecone 将越来越有能力完成一个经典超级计算机无法在合理时间内完成的任务。
|
||||||
|
|
||||||
@ -154,11 +153,9 @@ Cirq 将允许我们去:
|
|||||||
* 在设备上放置适当的门
|
* 在设备上放置适当的门
|
||||||
* 并调度这个门的时刻
|
* 并调度这个门的时刻
|
||||||
|
|
||||||
|
### 开放科学关于 Cirq 的观点
|
||||||
|
|
||||||
|
我们知道 Cirq 是在 GitHub 上开源的,在开源科学社区之外,特别是那些专注于量子研究的人们,都可以通过高效率地合作,通过开发新方法,去降低现有量子模型中的错误率和提升精确度,以解决目前在量子计算中所面临的挑战。
|
||||||
### Open Science 关于 Cirq 的观点
|
|
||||||
|
|
||||||
我们知道 Cirq 是在 GitHub 上开源的,它除了在 Open Science 社区之外,特别是那些专注于量子研究的人们,都可以高效率地合作,通过开发新方法,去降低现有量子模型中的错误率和提升精确度,以解决目前在量子计算中所面临的挑战。
|
|
||||||
|
|
||||||
如果 Cirq 不走开源模型的路线,事情可能变得更具挑战。一个伟大的创举可能就此错过,我们可能在量子计算领域止步不前。
|
如果 Cirq 不走开源模型的路线,事情可能变得更具挑战。一个伟大的创举可能就此错过,我们可能在量子计算领域止步不前。
|
||||||
|
|
||||||
@ -170,7 +167,7 @@ Cirq 将允许我们去:
|
|||||||
|
|
||||||
最后,我们看了两个示例 OpenFermion 和 Bristlecone,介绍了在量子计算中,Cirq 在开发研究中具有什么样的基本优势。最后我们以 Open Science 社区的视角对 Cirq 进行了一些精彩的思考,结束了我们的话题。
|
最后,我们看了两个示例 OpenFermion 和 Bristlecone,介绍了在量子计算中,Cirq 在开发研究中具有什么样的基本优势。最后我们以 Open Science 社区的视角对 Cirq 进行了一些精彩的思考,结束了我们的话题。
|
||||||
|
|
||||||
我们希望能以一种易于理解的方式向你介绍量子计算框架 Cirq 的使用。如果你有与此相关的任何反馈,请在下面的评论区告诉我们。感谢阅读,希望我们能在 Open Science 的下一篇文章中再见。
|
我们希望能以一种易于理解的方式向你介绍量子计算框架 Cirq 的使用。如果你有与此相关的任何反馈,请在下面的评论区告诉我们。感谢阅读,希望我们能在开放科学栏目的下一篇文章中再见。
|
||||||
|
|
||||||
--------------------------------------------------------------------------------
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
@ -179,7 +176,7 @@ via: https://itsfoss.com/qunatum-computing-cirq-framework/
|
|||||||
作者:[Avimanyu Bandyopadhyay][a]
|
作者:[Avimanyu Bandyopadhyay][a]
|
||||||
选题:[lujun9972](https://github.com/lujun9972)
|
选题:[lujun9972](https://github.com/lujun9972)
|
||||||
译者:[qhwdw](https://github.com/qhwdw)
|
译者:[qhwdw](https://github.com/qhwdw)
|
||||||
校对:[校对者ID](https://github.com/校对者ID)
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
@ -188,7 +185,7 @@ via: https://itsfoss.com/qunatum-computing-cirq-framework/
|
|||||||
[2]:https://en.wikipedia.org/wiki/Quantum
|
[2]:https://en.wikipedia.org/wiki/Quantum
|
||||||
[3]:https://www.clerro.com/guide/491/quantum-superposition-and-entanglement-explained
|
[3]:https://www.clerro.com/guide/491/quantum-superposition-and-entanglement-explained
|
||||||
[4]:https://physics.stackexchange.com/questions/148131/can-quantum-entanglement-and-quantum-superposition-be-considered-the-same-phenom
|
[4]:https://physics.stackexchange.com/questions/148131/can-quantum-entanglement-and-quantum-superposition-be-considered-the-same-phenom
|
||||||
[5]:https://4bds6hergc-flywheel.netdna-ssl.com/wp-content/uploads/2018/08/bit-vs-qubit.jpg
|
[5]:https://i0.wp.com/itsfoss.com/wp-content/uploads/2018/08/bit-vs-qubit.jpg?w=576&ssl=1
|
||||||
[6]:http://www.rfwireless-world.com/Terminology/Difference-between-Bit-and-Qubit.html
|
[6]:http://www.rfwireless-world.com/Terminology/Difference-between-Bit-and-Qubit.html
|
||||||
[7]:https://en.wikipedia.org/wiki/Proof_by_exhaustion
|
[7]:https://en.wikipedia.org/wiki/Proof_by_exhaustion
|
||||||
[8]:https://www.explainthatstuff.com/how-supercomputers-work.html
|
[8]:https://www.explainthatstuff.com/how-supercomputers-work.html
|
||||||
@ -199,11 +196,11 @@ via: https://itsfoss.com/qunatum-computing-cirq-framework/
|
|||||||
[13]:https://itsfoss.com/python-setup-linux/
|
[13]:https://itsfoss.com/python-setup-linux/
|
||||||
[14]:https://virtualenv.pypa.io
|
[14]:https://virtualenv.pypa.io
|
||||||
[15]:https://cirq.readthedocs.io/en/latest/install.html#installing-on-linux
|
[15]:https://cirq.readthedocs.io/en/latest/install.html#installing-on-linux
|
||||||
[16]:https://4bds6hergc-flywheel.netdna-ssl.com/wp-content/uploads/2018/08/cirq-framework-linux.jpeg
|
[16]:https://i0.wp.com/itsfoss.com/wp-content/uploads/2018/08/cirq-framework-linux.jpeg
|
||||||
[17]:https://pypi.org/project/pip/
|
[17]:https://pypi.org/project/pip/
|
||||||
[18]:https://4bds6hergc-flywheel.netdna-ssl.com/wp-content/uploads/2018/08/cirq-test-output.jpg
|
[18]:https://i0.wp.com/itsfoss.com/wp-content/uploads/2018/08/cirq-test-output.jpg
|
||||||
[19]:https://itsfoss.com/install-pycharm-ubuntu/
|
[19]:https://itsfoss.com/install-pycharm-ubuntu/
|
||||||
[20]:https://4bds6hergc-flywheel.netdna-ssl.com/wp-content/uploads/2018/08/cirq-tested-on-pycharm.jpg
|
[20]:https://i0.wp.com/itsfoss.com/wp-content/uploads/2018/08/cirq-tested-on-pycharm.jpg
|
||||||
[21]:https://github.com/quantumlib/Cirq/tree/master/examples
|
[21]:https://github.com/quantumlib/Cirq/tree/master/examples
|
||||||
[22]:https://github.com/quantumlib/Cirq/blob/master/docs/tutorial.md
|
[22]:https://github.com/quantumlib/Cirq/blob/master/docs/tutorial.md
|
||||||
[23]:http://mmrc.amss.cas.cn/tlb/201702/W020170224608149940643.pdf
|
[23]:http://mmrc.amss.cas.cn/tlb/201702/W020170224608149940643.pdf
|
@ -1,116 +1,105 @@
|
|||||||
# 如何使用 Fedora IoT 开启 LED 灯
|
如何使用 Fedora IoT 点亮 LED 灯
|
||||||
|
======
|
||||||
|
|
||||||
![](https://fedoramagazine.org/wp-content/uploads/2018/08/LED-IoT-816x345.jpg)
|
![](https://fedoramagazine.org/wp-content/uploads/2018/08/LED-IoT-816x345.jpg)
|
||||||
|
|
||||||
你喜欢 Fedora、容器和树莓派吗?这三者结合操控 LED 会怎么样?本文介绍的是 Fedora IoT,将展示如何在树莓派上安装预览镜像。还将学习如何与 GPIO 交互以开启 LED。
|
如果你喜欢 Fedora、容器,而且有一块树莓派,那么这三者结合操控 LED 会怎么样?本文介绍的是 Fedora IoT,将展示如何在树莓派上安装预览镜像。还将学习如何与 GPIO 交互以点亮 LED。
|
||||||
|
|
||||||
### 什么是 Fedora IoT?
|
### 什么是 Fedora IoT?
|
||||||
|
|
||||||
Fedora IoT 是当前 Fedora 项目的目标之一,计划成为一个完整的 Fedora 版本。Fedora IoT 将是一个在ARM(目前仅限 aarch64)例如树莓派,以及 x86_64 架构设备上运行的系统。
|
Fedora IoT 是当前 Fedora 项目的目标之一,计划成为一个完整的 Fedora 版本。Fedora IoT 将是一个在 ARM(目前仅限 aarch64)设备上(例如树莓派),以及 x86_64 架构上运行的系统。
|
||||||
|
|
||||||
![][1]
|
![][1]
|
||||||
|
|
||||||
Fedora IoT 基于 OSTree 开发, 就像[Fedora Silverblue][2] 和以往的 [Atomic Host][3].
|
Fedora IoT 基于 OSTree 开发,就像 [Fedora Silverblue][2] 和以往的 [Atomic Host][3]。
|
||||||
|
|
||||||
### 下载和安装 Fedora IoT
|
### 下载和安装 Fedora IoT
|
||||||
|
|
||||||
官方 Fedora IoT 镜像将和 Fedora 29 一起发布。但是在此期间你可以下载 [Fedora 28-based 镜像][4] 来进行这个实验。
|
官方 Fedora IoT 镜像将和 Fedora 29 一起发布。但是在此期间你可以下载 [基于 Fedora 28 的镜像][4] 来进行这个实验。(LCTT 译注:截止至本译文发布,[Fedora 29 已经发布了][11],但是 IoT 版本并未随同发布,或许会在 Fedora 30 一同发布?)
|
||||||
|
|
||||||
你有两种方法来安装这个系统:使用 dd 命令闪存SD卡,或者使用 fedora-arm-installer 工具。Fedora 的 Wiki 里面提供了更多关于[设置物理设备][5] 的信息来开发 IoT。另外,你可能需要调整第三个分区的大小。
|
你有两种方法来安装这个系统:要么使用 `dd` 命令烧录 SD 卡,或者使用 `fedora-arm-installer` 工具。Fedora 的 Wiki 里面提供了为 IoT [设置物理设备][5] 的更多信息。另外,你可能需要调整第三个分区的大小。
|
||||||
|
|
||||||
把 SD 卡插入到设备并运行,需要创建一个用户来完成安装。这个步骤需要串行连接或带键盘的 HDMI 显示器来与设备进行交互。
|
把 SD 卡插入到设备后,你需要创建一个用户来完成安装。这个步骤需要串行连接或一个 HDMI 显示器和键盘来与设备进行交互。
|
||||||
|
|
||||||
当系统安装完成后,下一步就是要设置网络连接。使用你刚才创建的用户登录系统,可以使用下列方式之一完成网络连接设置:
|
当系统安装完成后,下一步就是要设置网络连接。使用你刚才创建的用户登录系统,可以使用下列方式之一完成网络连接设置:
|
||||||
|
|
||||||
- 如果你需要手动配置你的网络,可能需要执行类似如下命令,需要保证设置正确的网络地址:
|
* 如果你需要手动配置你的网络,可能需要执行类似如下命令,需要保证设置正确的网络地址:
|
||||||
|
|
||||||
```
|
```
|
||||||
$ nmcli connection add con-name cable ipv4.addresses \
|
$ nmcli connection add con-name cable ipv4.addresses \
|
||||||
192.168.0.10/24 ipv4.gateway 192.168.0.1 \
|
192.168.0.10/24 ipv4.gateway 192.168.0.1 \
|
||||||
connection.autoconnect true ipv4.dns "8.8.8.8,1.1.1.1" \
|
connection.autoconnect true ipv4.dns "8.8.8.8,1.1.1.1" \
|
||||||
type ethernet ifname eth0 ipv4.method manual
|
type ethernet ifname eth0 ipv4.method manual
|
||||||
|
|
||||||
```
|
```
|
||||||
|
|
||||||
- 如果你网络上运行着 DHCP 服务,可能需要类似如下命令:
|
* 如果你网络上运行着 DHCP 服务,可能需要类似如下命令:
|
||||||
|
|
||||||
```
|
```
|
||||||
$ nmcli con add type ethernet con-name cable ifname eth0
|
$ nmcli con add type ethernet con-name cable ifname eth0
|
||||||
```
|
```
|
||||||
|
|
||||||
|
### Fedora 中的 GPIO 接口
|
||||||
|
|
||||||
### **Fedora 中的 GPIO 接口**
|
|
||||||
|
|
||||||
许多关于 Linux 上 GPIO 的教程都关注传统的 GPIO sysfis 接口。这个接口已经不推荐使用了,并且上游 Linux 内核社区由于安全和其他问题的缘故打算完全删除它。
|
许多关于 Linux 上 GPIO 的教程都关注传统的 GPIO sysfis 接口。这个接口已经不推荐使用了,并且上游 Linux 内核社区由于安全和其他问题的缘故打算完全删除它。
|
||||||
|
|
||||||
Fedora 已经不将这个传统的接口编译到内核了,因此在系统上没有 /sys/class/gpio 这个文件。此教程使用一个上游内核提供的一个新的字符设备 /dev/gpiochipN 。这是目前和 GPIO 交互的方式。
|
Fedora 已经不将这个传统的接口编译到内核了,因此在系统上没有 `/sys/class/gpio` 这个文件。此教程使用一个上游内核提供的一个新的字符设备 `/dev/gpiochipN` 。这是目前和 GPIO 交互的方式。
|
||||||
|
|
||||||
为了和这个新设备进行交互,你需要使用一个库和一系列命令行界面工具。公共命令行工具比如说 echo 和 cat 在此设备上无法正常工作。
|
为了和这个新设备进行交互,你需要使用一个库和一系列命令行界面的工具。常用的命令行工具比如说 `echo` 和 `cat` 在此设备上无法正常工作。
|
||||||
|
|
||||||
你可以通过安装 libgpiod-utils 包来安装命令行界面工具。python3-libgpiod 包提供了相应的 Python 库。
|
你可以通过安装 libgpiod-utils 包来安装命令行界面工具。python3-libgpiod 包提供了相应的 Python 库。
|
||||||
|
|
||||||
### **使用 Podman 来创建一个容器**
|
### 使用 Podman 来创建一个容器
|
||||||
|
|
||||||
[Podman][6] 是一个容器运行环境,其命令行界面类似于Docker。Podman的一大优势是它不会在后台运行任何守护进程。这对于资源有限的设备尤其有用。Podman 还允许您使用 systemd 单元文件启动容器化服务。此外,它还有许多其他功能。
|
[Podman][6] 是一个容器运行环境,其命令行界面类似于 Docker。Podman 的一大优势是它不会在后台运行任何守护进程。这对于资源有限的设备尤其有用。Podman 还允许您使用 systemd 单元文件启动容器化服务。此外,它还有许多其他功能。
|
||||||
|
|
||||||
我们使用如下两步来创建一个容器:
|
我们使用如下两步来创建一个容器:
|
||||||
|
|
||||||
```
|
|
||||||
1. 创建包含所需包的分层镜像。
|
1. 创建包含所需包的分层镜像。
|
||||||
2. 使用分层镜像创建一个新容器。
|
2. 使用分层镜像创建一个新容器。
|
||||||
```
|
|
||||||
|
|
||||||
|
首先创建一个 Dockerfile 文件,内容如下。这些内容告诉 Podman 基于可使用的最新 Fedora 镜像来构建我们的分层镜像。然后就是更新系统和安装一些软件包:
|
||||||
|
|
||||||
首先创建一个 Dockerfile 文件,内容如下。这些内容告诉 podman 基于可使用的最新 Fedora 镜像来构建我们的分层镜像。然后就是更新系统和安装一些软件包:
|
|
||||||
|
|
||||||
```
|
```
|
||||||
FROM fedora:latest
|
FROM fedora:latest
|
||||||
RUN dnf -y update
|
RUN dnf -y update
|
||||||
RUN dnf -y install libgpiod-utils python3-libgpiod
|
RUN dnf -y install libgpiod-utils python3-libgpiod
|
||||||
|
|
||||||
```
|
```
|
||||||
|
|
||||||
这样你就完成了镜像的生成前的配置工作,这个镜像基于最新的 Fedora,而且包含了和 GPIO 交互的软件包。
|
这样你就完成了镜像的生成前的配置工作,这个镜像基于最新的 Fedora,而且包含了和 GPIO 交互的软件包。
|
||||||
|
|
||||||
现在你就可以运行下方命令来构建你的基本镜像了:
|
现在你就可以运行如下命令来构建你的基本镜像了:
|
||||||
|
|
||||||
```
|
```
|
||||||
$ sudo podman build --tag fedora:gpiobase -f ./Dockerfile
|
$ sudo podman build --tag fedora:gpiobase -f ./Dockerfile
|
||||||
|
|
||||||
```
|
```
|
||||||
|
|
||||||
你已经成功创建了你的自定义镜像。这样以后你就可以不用每次都重新搭建环境了,而是基于你创建的镜像来完成工作。
|
你已经成功创建了你的自定义镜像。这样以后你就可以不用每次都重新搭建环境了,而是基于你创建的镜像来完成工作。
|
||||||
|
|
||||||
### 使用 Podman 完成工作
|
### 使用 Podman 完成工作
|
||||||
|
|
||||||
为了确认当前的镜像,可以运行下方命令:
|
为了确认当前的镜像是否就绪,可以运行如下命令:
|
||||||
|
|
||||||
```
|
```
|
||||||
$ sudo podman images
|
$ sudo podman images
|
||||||
REPOSITORY TAG IMAGE ID CREATED SIZE
|
REPOSITORY TAG IMAGE ID CREATED SIZE
|
||||||
localhost/fedora gpiobase 67a2b2b93b4b 10 minutes ago 488MB
|
localhost/fedora gpiobase 67a2b2b93b4b 10 minutes ago 488MB
|
||||||
docker.io/library/fedora latest c18042d7fac6 2 days ago 300MB
|
docker.io/library/fedora latest c18042d7fac6 2 days ago 300MB
|
||||||
|
|
||||||
```
|
```
|
||||||
|
|
||||||
现在,启动容器并进行一些实际的实验。 容器通常是隔离的,无法访问主机系统,包括GPIO接口。 因此需要在启动容器时将其挂载在容器内。 可以使用以下命令中的 -device 选项来解决:
|
现在,启动容器并进行一些实际的实验。容器通常是隔离的,无法访问主机系统,包括 GPIO 接口。因此需要在启动容器时将其挂载在容器内。可以使用以下命令中的 `-device` 选项来解决:
|
||||||
|
|
||||||
```
|
```
|
||||||
$ sudo podman run -it --name gpioexperiment --device=/dev/gpiochip0 localhost/fedora:gpiobase /bin/bash
|
$ sudo podman run -it --name gpioexperiment --device=/dev/gpiochip0 localhost/fedora:gpiobase /bin/bash
|
||||||
|
|
||||||
```
|
```
|
||||||
|
|
||||||
运行之后就进入了正在运行的容器中。 在继续之前,这里有一些容器命令。 输入 exit 或者按下 **Ctrl+D** 来退出容器。
|
运行之后就进入了正在运行的容器中。在继续之前,这里有一些容器命令。输入 `exit` 或者按下 `Ctrl+D` 来退出容器。
|
||||||
|
|
||||||
显示所有存在的容器可以运行如下命令:
|
显示所有存在的容器可以运行如下命令,这包括当前没有运行的,比如你刚刚创建的那个:
|
||||||
|
|
||||||
```
|
```
|
||||||
$ sudo podman container ls -a
|
$ sudo podman container ls -a
|
||||||
CONTAINER ID IMAGE COMMAND CREATED STATUS PORTS NAMES
|
CONTAINER ID IMAGE COMMAND CREATED STATUS PORTS NAMES
|
||||||
64e661d5d4e8 localhost/fedora:gpiobase /bin/bash 37 seconds ago Exited (0) Less than a second ago gpioexperiment
|
64e661d5d4e8 localhost/fedora:gpiobase /bin/bash 37 seconds ago Exited (0) Less than a second ago gpioexperiment
|
||||||
|
|
||||||
```
|
```
|
||||||
|
|
||||||
使用如下命令创建一个新的容器:
|
使用如下命令创建一个新的容器:
|
||||||
@ -127,55 +116,50 @@ $ sudo podman rm newexperiment
|
|||||||
|
|
||||||
```
|
```
|
||||||
|
|
||||||
### **开启 LED 灯**
|
### 点亮 LED 灯
|
||||||
|
|
||||||
现在可以使用已创建的容器。 如果容器已经退出,请使用以下命令再次启动它:
|
现在可以使用已创建的容器。如果已经从容器退出,请使用以下命令再次启动它:
|
||||||
|
|
||||||
```
|
```
|
||||||
$ sudo podman start -ia gpioexperiment
|
$ sudo podman start -ia gpioexperiment
|
||||||
|
|
||||||
```
|
```
|
||||||
|
|
||||||
如前所述,可以使用 Fedora 中 libgpiod-utils 包提供的 CLI 工具。 要列出可用的 GPIO 芯片可以使用如下命令:
|
如前所述,可以使用 Fedora 中 libgpiod-utils 包提供的命令行工具。要列出可用的 GPIO 芯片可以使用如下命令:
|
||||||
|
|
||||||
```
|
```
|
||||||
$ gpiodetect
|
$ gpiodetect
|
||||||
gpiochip0 [pinctrl-bcm2835] (54 lines)
|
gpiochip0 [pinctrl-bcm2835] (54 lines)
|
||||||
|
|
||||||
```
|
```
|
||||||
|
|
||||||
要获取特定芯片的公开列表,请运行:
|
要获取特定芯片的连线列表,请运行:
|
||||||
|
|
||||||
```
|
```
|
||||||
$ gpioinfo gpiochip0
|
$ gpioinfo gpiochip0
|
||||||
|
|
||||||
```
|
```
|
||||||
|
|
||||||
请注意,物理引脚数与前一个命令打印的行数之间没有相关性。 重要的是 BCM 编号,如 [pinout.xyz][7] 所示。 建议不要使用没有相应 BCM 编号的线路。
|
请注意,物理引脚数与前一个命令所打印的连线数之间没有相关性。重要的是 BCM 编号,如 [pinout.xyz][7] 所示。建议不要使用没有相应 BCM 编号的连线。
|
||||||
|
|
||||||
现在,将 LED 连接到物理引脚40,也就是 BCM 21。请记住:LED的短腿(负极,称为阴极)必须连接到带有330欧姆电阻的树莓派的 GND 引脚, 并且长腿(阳极)到物理引脚40。
|
现在,将 LED 连接到物理引脚 40,也就是 BCM 21。请记住:LED 的短腿(负极,称为阴极)必须连接到带有 330 欧姆电阻的树莓派的 GND 引脚, 并且长腿(阳极)到物理引脚 40。
|
||||||
|
|
||||||
运行以下命令打开LED。,按下 **Ctrl + C ** 关闭:
|
运行以下命令点亮 LED,按下 `Ctrl + C` 关闭:
|
||||||
|
|
||||||
```
|
```
|
||||||
$ gpioset --mode=wait gpiochip0 21=1
|
$ gpioset --mode=wait gpiochip0 21=1
|
||||||
|
|
||||||
```
|
```
|
||||||
|
|
||||||
要点亮一段时间,请添加 -b(在后台运行)和 -s NUM(多少秒)参数,如下所示。 例如,要点亮 LED 5秒钟,运行如下命令:
|
要点亮一段时间,请添加 `-b`(在后台运行)和 `-s NUM`(多少秒)参数,如下所示。 例如,要点亮 LED 5 秒钟,运行如下命令:
|
||||||
|
|
||||||
```
|
```
|
||||||
$ gpioset -b -s 5 --mode=time gpiochip0 21=1
|
$ gpioset -b -s 5 --mode=time gpiochip0 21=1
|
||||||
|
|
||||||
```
|
```
|
||||||
|
|
||||||
另一个有用的命令是 gpioget。 它可以获得引脚的状态(高或低),可用于检测按钮和开关。
|
另一个有用的命令是 `gpioget`。 它可以获得引脚的状态(高或低),可用于检测按钮和开关。
|
||||||
|
|
||||||
![Closeup of LED connection with GPIO][8]
|
![Closeup of LED connection with GPIO][8]
|
||||||
|
|
||||||
### **总结**
|
### 总结
|
||||||
|
|
||||||
你也可以使用 Python 操控 LED - [这里有一些例子][9]。 也可以在容器内使用 i2c 设备。 此外,Podman 与此 Fedora 版本并不严格相关。 你可以在任何现有的 Fedora Edition 上安装它,或者在 Fedora 中使用两个基于 OSTree 的新系统进行尝试:[Fedora Silverblue][2] 和 [Fedora CoreOS][10]。
|
你也可以使用 Python 操控 LED —— [这里有一些例子][9]。 也可以在容器内使用 i2c 设备。 此外,Podman 与此 Fedora 版本并不严格相关。你可以在任何现有的 Fedora 版本上安装它,或者在 Fedora 中使用两个基于 OSTree 的新系统进行尝试:[Fedora Silverblue][2] 和 [Fedora CoreOS][10]。
|
||||||
|
|
||||||
------
|
------
|
||||||
|
|
||||||
@ -184,7 +168,7 @@ via: https://fedoramagazine.org/turnon-led-fedora-iot/
|
|||||||
作者:[Alessio Ciregia][a]
|
作者:[Alessio Ciregia][a]
|
||||||
选题:[lujun9972](https://github.com/lujun9972)
|
选题:[lujun9972](https://github.com/lujun9972)
|
||||||
译者:[ScarboroughCoral](https://github.com/ScarboroughCoral)
|
译者:[ScarboroughCoral](https://github.com/ScarboroughCoral)
|
||||||
校对:[校对者ID](https://github.com/校对者ID)
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
@ -199,3 +183,4 @@ via: https://fedoramagazine.org/turnon-led-fedora-iot/
|
|||||||
[8]: https://fedoramagazine.org/wp-content/uploads/2018/08/breadboard-1024x768.png
|
[8]: https://fedoramagazine.org/wp-content/uploads/2018/08/breadboard-1024x768.png
|
||||||
[9]: https://github.com/brgl/libgpiod/tree/master/bindings/python/examples
|
[9]: https://github.com/brgl/libgpiod/tree/master/bindings/python/examples
|
||||||
[10]: https://coreos.fedoraproject.org/
|
[10]: https://coreos.fedoraproject.org/
|
||||||
|
[11]: https://fedoramagazine.org/announcing-fedora-29/
|
@ -1,4 +1,4 @@
|
|||||||
实验 5:文件系统、Spawn 和 Shell
|
Caffeinated 6.828:实验 5:文件系统、Spawn 和 Shell
|
||||||
======
|
======
|
||||||
|
|
||||||
### 简介
|
### 简介
|
||||||
@ -10,31 +10,31 @@
|
|||||||
使用 Git 去获取最新版的课程仓库,然后创建一个命名为 `lab5` 的本地分支,去跟踪远程的 `origin/lab5` 分支:
|
使用 Git 去获取最新版的课程仓库,然后创建一个命名为 `lab5` 的本地分支,去跟踪远程的 `origin/lab5` 分支:
|
||||||
|
|
||||||
```
|
```
|
||||||
athena% cd ~/6.828/lab
|
athena% cd ~/6.828/lab
|
||||||
athena% add git
|
athena% add git
|
||||||
athena% git pull
|
athena% git pull
|
||||||
Already up-to-date.
|
Already up-to-date.
|
||||||
athena% git checkout -b lab5 origin/lab5
|
athena% git checkout -b lab5 origin/lab5
|
||||||
Branch lab5 set up to track remote branch refs/remotes/origin/lab5.
|
Branch lab5 set up to track remote branch refs/remotes/origin/lab5.
|
||||||
Switched to a new branch "lab5"
|
Switched to a new branch "lab5"
|
||||||
athena% git merge lab4
|
athena% git merge lab4
|
||||||
Merge made by recursive.
|
Merge made by recursive.
|
||||||
.....
|
.....
|
||||||
athena%
|
athena%
|
||||||
```
|
```
|
||||||
|
|
||||||
在实验中这一部分的主要新组件是文件系统环境,它位于新的 `fs` 目录下。通过检查这个目录中的所有文件,我们来看一下新的文件都有什么。另外,在 `user` 和 `lib` 目录下还有一些文件系统相关的源文件。
|
在实验中这一部分的主要新组件是文件系统环境,它位于新的 `fs` 目录下。通过检查这个目录中的所有文件,我们来看一下新的文件都有什么。另外,在 `user` 和 `lib` 目录下还有一些文件系统相关的源文件。
|
||||||
|
|
||||||
fs/fs.c 维护文件系统在磁盘上结构的代码
|
- `fs/fs.c` 维护文件系统在磁盘上结构的代码
|
||||||
fs/bc.c 构建在我们的用户级页故障处理功能之上的一个简单的块缓存
|
- `fs/bc.c` 构建在我们的用户级页故障处理功能之上的一个简单的块缓存
|
||||||
fs/ide.c 极简的基于 PIO(非中断驱动的)IDE 驱动程序代码
|
- `fs/ide.c` 极简的基于 PIO(非中断驱动的)IDE 驱动程序代码
|
||||||
fs/serv.c 使用文件系统 IPC 与客户端环境交互的文件系统服务器
|
- `fs/serv.c` 使用文件系统 IPC 与客户端环境交互的文件系统服务器
|
||||||
lib/fd.c 实现一个常见的类 UNIX 的文件描述符接口的代码
|
- `lib/fd.c` 实现一个常见的类 UNIX 的文件描述符接口的代码
|
||||||
lib/file.c 磁盘上文件类型的驱动,实现为一个文件系统 IPC 客户端
|
- `lib/file.c` 磁盘上文件类型的驱动,实现为一个文件系统 IPC 客户端
|
||||||
lib/console.c 控制台输入/输出文件类型的驱动
|
- `lib/console.c` 控制台输入/输出文件类型的驱动
|
||||||
lib/spawn.c spawn 库调用的框架代码
|
- `lib/spawn.c` spawn 库调用的框架代码
|
||||||
|
|
||||||
你应该再次去运行 `pingpong`、`primes`、和 `forktree`,测试实验 4 完成后合并到新的实验 5 中的代码能否正确运行。你还需要在 `kern/init.c` 中注释掉 `ENV_CREATE(fs_fs)` 行,因为 `fs/fs.c` 将尝试去做一些 I/O,而 JOS 到目前为止还不具备该功能。同样,在 `lib/exit.c` 中临时注释掉对 `close_all()` 的调用;这个函数将调用你在本实验后面部分去实现的子程序,如果现在去调用,它将导致 JOS 内核崩溃。如果你的实验 4 的代码没有任何 bug,将很完美地通过这个测试。在它们都能正常工作之前是不能继续后续实验的。在你开始做练习 1 时,不要忘记去取消这些行上的注释。
|
你应该再次去运行 `pingpong`、`primes` 和 `forktree`,测试实验 4 完成后合并到新的实验 5 中的代码能否正确运行。你还需要在 `kern/init.c` 中注释掉 `ENV_CREATE(fs_fs)` 行,因为 `fs/fs.c` 将尝试去做一些 I/O,而 JOS 到目前为止还不具备该功能。同样,在 `lib/exit.c` 中临时注释掉对 `close_all()` 的调用;这个函数将调用你在本实验后面部分去实现的子程序,如果现在去调用,它将导致 JOS 内核崩溃。如果你的实验 4 的代码没有任何 bug,将很完美地通过这个测试。在它们都能正常工作之前是不能继续后续实验的。在你开始做练习 1 时,不要忘记去取消这些行上的注释。
|
||||||
|
|
||||||
如果它们不能正常工作,使用 `git diff lab4` 去重新评估所有的变更,确保你在实验 4(及以前)所写的代码在本实验中没有丢失。确保实验 4 仍然能正常工作。
|
如果它们不能正常工作,使用 `git diff lab4` 去重新评估所有的变更,确保你在实验 4(及以前)所写的代码在本实验中没有丢失。确保实验 4 仍然能正常工作。
|
||||||
|
|
||||||
@ -44,11 +44,11 @@ lib/spawn.c spawn 库调用的框架代码
|
|||||||
|
|
||||||
### 文件系统的雏形
|
### 文件系统的雏形
|
||||||
|
|
||||||
你将要使用的文件系统比起大多数“真正的”文件系统(包括 xv6 UNIX 的文件系统)要简单的多,但它也是很强大的,足够去提供基本的特性:创建、读取、写入、和删除组织在层次目录结构中的文件。
|
你将要使用的文件系统比起大多数“真正的”文件系统(包括 xv6 UNIX 的文件系统)要简单的多,但它也是很强大的,足够去提供基本的特性:创建、读取、写入和删除组织在层次目录结构中的文件。
|
||||||
|
|
||||||
到目前为止,我们开发的是一个单用户操作系统,它提供足够的保护并能去捕获 bug,但它还不能在多个不可信用户之间提供保护。因此,我们的文件系统还不支持 UNIX 的所有者或权限的概念。我们的文件系统目前也不支持硬链接、时间戳、或像大多数 UNIX 文件系统实现的那些特殊的设备文件。
|
到目前为止,我们开发的是一个单用户操作系统,它提供足够的保护并能去捕获 bug,但它还不能在多个不可信用户之间提供保护。因此,我们的文件系统还不支持 UNIX 的所有者或权限的概念。我们的文件系统目前也不支持硬链接、时间戳、或像大多数 UNIX 文件系统实现的那些特殊的设备文件。
|
||||||
|
|
||||||
### 磁盘上文件系统结构
|
### 磁盘上的文件系统结构
|
||||||
|
|
||||||
主流的 UNIX 文件系统将可用磁盘空间分为两种主要的区域类型:节点区域和数据区域。UNIX 文件系统在文件系统中为每个文件分配一个节点;一个文件的节点保存了这个文件重要的元数据,比如它的 `stat` 属性和指向数据块的指针。数据区域被分为更大的(一般是 8 KB 或更大)数据块,它在文件系统中存储文件数据和目录元数据。目录条目包含文件名字和指向到节点的指针;如果文件系统中的多个目录条目指向到那个文件的节点上,则称该文件是硬链接的。由于我们的文件系统不支持硬链接,所以我们不需要这种间接的级别,并且因此可以更方便简化:我们的文件系统将压根就不使用节点,而是简单地将文件的(或子目录的)所有元数据保存在描述那个文件的(唯一的)目录条目中。
|
主流的 UNIX 文件系统将可用磁盘空间分为两种主要的区域类型:节点区域和数据区域。UNIX 文件系统在文件系统中为每个文件分配一个节点;一个文件的节点保存了这个文件重要的元数据,比如它的 `stat` 属性和指向数据块的指针。数据区域被分为更大的(一般是 8 KB 或更大)数据块,它在文件系统中存储文件数据和目录元数据。目录条目包含文件名字和指向到节点的指针;如果文件系统中的多个目录条目指向到那个文件的节点上,则称该文件是硬链接的。由于我们的文件系统不支持硬链接,所以我们不需要这种间接的级别,并且因此可以更方便简化:我们的文件系统将压根就不使用节点,而是简单地将文件的(或子目录的)所有元数据保存在描述那个文件的(唯一的)目录条目中。
|
||||||
|
|
||||||
@ -71,6 +71,7 @@ UNIX xv6 文件系统使用 512 字节大小的块,与它底层磁盘的扇区
|
|||||||
#### 文件元数据
|
#### 文件元数据
|
||||||
|
|
||||||
![File structure][2]
|
![File structure][2]
|
||||||
|
|
||||||
元数据的布局是描述在我们的文件系统中的一个文件中,这个文件就是 `inc/fs.h` 中的 `struct File`。元数据包含文件的名字、大小、类型(普通文件还是目录)、指向构成这个文件的块的指针。正如前面所提到的,我们的文件系统中并没有节点,因此元数据是保存在磁盘上的一个目录条目中,而不是像大多数“真正的”文件系统那样保存在节点中。为简单起见,我们将使用 `File` 这一个结构去表示文件元数据,因为它要同时出现在磁盘上和内存中。
|
元数据的布局是描述在我们的文件系统中的一个文件中,这个文件就是 `inc/fs.h` 中的 `struct File`。元数据包含文件的名字、大小、类型(普通文件还是目录)、指向构成这个文件的块的指针。正如前面所提到的,我们的文件系统中并没有节点,因此元数据是保存在磁盘上的一个目录条目中,而不是像大多数“真正的”文件系统那样保存在节点中。为简单起见,我们将使用 `File` 这一个结构去表示文件元数据,因为它要同时出现在磁盘上和内存中。
|
||||||
|
|
||||||
在 `struct File` 中的数组 `f_direct` 包含一个保存文件的前 10 个块(`NDIRECT`)的块编号的空间,我们称之为文件的直接块。对于最大 `10*4096 = 40KB` 的小文件,这意味着这个文件的所有块的块编号将全部直接保存在结构 `File` 中,但是,对于超过 40 KB 大小的文件,我们需要一个地方去保存文件剩余的块编号。所以我们分配一个额外的磁盘块,我们称之为文件的间接块,由它去保存最多 4096/4 = 1024 个额外的块编号。因此,我们的文件系统最多允许有 1034 个块,或者说不能超过 4MB 大小。为支持大文件,“真正的”文件系统一般都支持两个或三个间接块。
|
在 `struct File` 中的数组 `f_direct` 包含一个保存文件的前 10 个块(`NDIRECT`)的块编号的空间,我们称之为文件的直接块。对于最大 `10*4096 = 40KB` 的小文件,这意味着这个文件的所有块的块编号将全部直接保存在结构 `File` 中,但是,对于超过 40 KB 大小的文件,我们需要一个地方去保存文件剩余的块编号。所以我们分配一个额外的磁盘块,我们称之为文件的间接块,由它去保存最多 4096/4 = 1024 个额外的块编号。因此,我们的文件系统最多允许有 1034 个块,或者说不能超过 4MB 大小。为支持大文件,“真正的”文件系统一般都支持两个或三个间接块。
|
||||||
@ -91,33 +92,28 @@ UNIX xv6 文件系统使用 512 字节大小的块,与它底层磁盘的扇区
|
|||||||
|
|
||||||
只要我们依赖轮询、基于 “编程的 I/O”(PIO)的磁盘访问,并且不使用磁盘中断,那么在用户空间中实现磁盘访问还是很容易的。也可以去实现由中断驱动的设备驱动程序(比如像 L3 和 L4 内核就是这么做的),但这样做的话,内核必须接收设备中断并将它派发到相应的用户模式环境上,这样实现的难度会更大。
|
只要我们依赖轮询、基于 “编程的 I/O”(PIO)的磁盘访问,并且不使用磁盘中断,那么在用户空间中实现磁盘访问还是很容易的。也可以去实现由中断驱动的设备驱动程序(比如像 L3 和 L4 内核就是这么做的),但这样做的话,内核必须接收设备中断并将它派发到相应的用户模式环境上,这样实现的难度会更大。
|
||||||
|
|
||||||
x86 处理器在 EFLAGS 寄存器中使用 IOPL 位去确定保护模式中的代码是否允许执行特定的设备 I/O 指令,比如 `IN` 和 `OUT` 指令。由于我们需要的所有 IDE 磁盘寄存器都位于 x86 的 I/O 空间中而不是映射在内存中,所以,为了允许文件系统去访问这些寄存器,我们需要做的唯一的事情便是授予文件系统环境“I/O 权限”。实际上,在 EFLAGS 寄存器的 IOPL 位上规定,内核使用一个简单的“要么全都能访问、要么全都不能访问”的方法来控制用户模式中的代码能否访问 I/O 空间。在我们的案例中,我们希望文件系统环境能够去访问 I/O 空间,但我们又希望任何其它的环境完全不能访问 I/O 空间。
|
x86 处理器在 EFLAGS 寄存器中使用 IOPL 位去确定保护模式中的代码是否允许执行特定的设备 I/O 指令,比如 `IN` 和 `OUT` 指令。由于我们需要的所有 IDE 磁盘寄存器都位于 x86 的 I/O 空间中而不是映射在内存中,所以,为了允许文件系统去访问这些寄存器,我们需要做的唯一的事情便是授予文件系统环境“I/O 权限”。实际上,在 EFLAGS 寄存器的 IOPL 位上规定,内核使用一个简单的“要么全都能访问、要么全都不能访问”的方法来控制用户模式中的代码能否访问 I/O 空间。在我们的案例中,我们希望文件系统环境能够去访问 I/O 空间,但我们又希望任何其它的环境完全不能访问 I/O 空间。
|
||||||
|
|
||||||
```markdown
|
> **练习 1**、`i386_init` 通过将类型 `ENV_TYPE_FS` 传递给你的环境创建函数 `env_create` 来识别文件系统。修改 `env.c` 中的 `env_create` ,以便于它只授予文件系统环境 I/O 的权限,而不授予任何其它环境 I/O 的权限。
|
||||||
练习 1、`i386_init` 通过将类型 `ENV_TYPE_FS` 传递给你的环境创建函数 `env_create` 来识别文件系统。修改 `env.c` 中的 `env_create` ,以便于它只授予文件系统环境 I/O 的权限,而不授予任何其它环境 I/O 的权限。
|
|
||||||
|
|
||||||
确保你能启动这个文件系统环境,而不会产生一般保护故障。你应该要通过在 `make grade` 中的 "fs i/o" 测试。
|
> 确保你能启动这个文件系统环境,而不会产生一般保护故障。你应该要通过在 `make grade` 中的 "fs i/o" 测试。
|
||||||
```
|
|
||||||
|
|
||||||
```markdown
|
.
|
||||||
问题
|
|
||||||
|
|
||||||
1、当你从一个环境切换到另一个环境时,你是否需要做一些操作来确保 I/O 权限设置能被保存和正确地恢复?为什么?
|
|
||||||
```
|
|
||||||
|
|
||||||
|
> **问题 1**、当你从一个环境切换到另一个环境时,你是否需要做一些操作来确保 I/O 权限设置能被保存和正确地恢复?为什么?
|
||||||
|
|
||||||
注意本实验中的 `GNUmakefile` 文件,它用于设置 QEMU 去使用文件 `obj/kern/kernel.img` 作为磁盘 0 的镜像(一般情况下表示 DOS 或 Windows 中的 “C 盘”),以及使用(新)文件 `obj/fs/fs.img` 作为磁盘 1 的镜像(”D 盘“)。在本实验中,我们的文件系统应该仅与磁盘 1 有交互;而磁盘 0 仅用于去引导内核。如果你想去恢复其中一个有某些错误的磁盘镜像,你可以通过输入如下的命令,去重置它们到最初的、”崭新的“版本:
|
注意本实验中的 `GNUmakefile` 文件,它用于设置 QEMU 去使用文件 `obj/kern/kernel.img` 作为磁盘 0 的镜像(一般情况下表示 DOS 或 Windows 中的 “C 盘”),以及使用(新)文件 `obj/fs/fs.img` 作为磁盘 1 的镜像(”D 盘“)。在本实验中,我们的文件系统应该仅与磁盘 1 有交互;而磁盘 0 仅用于去引导内核。如果你想去恢复其中一个有某些错误的磁盘镜像,你可以通过输入如下的命令,去重置它们到最初的、”崭新的“版本:
|
||||||
|
|
||||||
```
|
```
|
||||||
$ rm obj/kern/kernel.img obj/fs/fs.img
|
$ rm obj/kern/kernel.img obj/fs/fs.img
|
||||||
$ make
|
$ make
|
||||||
```
|
```
|
||||||
|
|
||||||
或者:
|
或者:
|
||||||
|
|
||||||
```
|
```
|
||||||
$ make clean
|
$ make clean
|
||||||
$ make
|
$ make
|
||||||
```
|
```
|
||||||
|
|
||||||
小挑战!实现中断驱动的 IDE 磁盘访问,既可以使用也可以不使用 DMA 模式。由你来决定是否将设备驱动移植进内核中、还是与文件系统一样保留在用户空间中、甚至是将它移植到一个它自己的的单独的环境中(如果你真的想了解微内核的本质的话)。
|
小挑战!实现中断驱动的 IDE 磁盘访问,既可以使用也可以不使用 DMA 模式。由你来决定是否将设备驱动移植进内核中、还是与文件系统一样保留在用户空间中、甚至是将它移植到一个它自己的的单独的环境中(如果你真的想了解微内核的本质的话)。
|
||||||
@ -132,45 +128,39 @@ x86 处理器在 EFLAGS 寄存器中使用 IOPL 位去确定保护模式中的
|
|||||||
|
|
||||||
当然,将整个磁盘读入到内存中需要很长时间,因此,我们将它实现成”按需“分页的形式,那样我们只在磁盘映射区域中分配页,并且当在这个区域中产生页故障时,从磁盘读取相关的块去响应这个页故障。通过这种方式,我们能够假装将整个磁盘装进了内存中。
|
当然,将整个磁盘读入到内存中需要很长时间,因此,我们将它实现成”按需“分页的形式,那样我们只在磁盘映射区域中分配页,并且当在这个区域中产生页故障时,从磁盘读取相关的块去响应这个页故障。通过这种方式,我们能够假装将整个磁盘装进了内存中。
|
||||||
|
|
||||||
```markdown
|
> **练习 2**、在 `fs/bc.c` 中实现 `bc_pgfault` 和 `flush_block` 函数。`bc_pgfault` 函数是一个页故障服务程序,就像你在前一个实验中编写的写时复制 fork 一样,只不过它的任务是从磁盘中加载页去响应一个页故障。在你编写它时,记住: (1) `addr` 可能并不会做边界对齐,并且 (2) 在扇区中的 `ide_read` 操作并不是以块为单位的。
|
||||||
练习 2、在 `fs/bc.c` 中实现 `bc_pgfault` 和 `flush_block` 函数。`bc_pgfault` 函数是一个页故障服务程序,就像你在前一个实验中编写的写时复制 fork 一样,只不过它的任务是从磁盘中加载页去响应一个页故障。在你编写它时,记住: (1) `addr` 可能并不会做边界对齐,并且 (2) 在扇区中的 `ide_read` 操作并不是以块为单位的。
|
|
||||||
|
|
||||||
(如果需要的话)函数 `flush_block` 应该会将一个块写入到磁盘上。如果在块缓存中没有块(也就是说,页没有映射)或者它不是一个脏块,那么 `flush_block` 将什么都不做。我们将使用虚拟内存硬件去跟踪,磁盘块自最后一次从磁盘读取或写入到磁盘之后是否被修改过。查看一个块是否需要写入时,我们只需要去查看 `uvpt` 条目中的 `PTE_D` 的 ”dirty“ 位即可。(`PTE_D` 位由处理器设置,用于表示那个页被写入;具体细节可以查看 x386 参考手册的 [第 5 章][3] 的 5.2.4.3 节)块被写入到磁盘上之后,`flush_block` 函数将使用 `sys_page_map` 去清除 `PTE_D` 位。
|
>(如果需要的话)函数 `flush_block` 应该会将一个块写入到磁盘上。如果在块缓存中没有块(也就是说,页没有映射)或者它不是一个脏块,那么 `flush_block` 将什么都不做。我们将使用虚拟内存硬件去跟踪,磁盘块自最后一次从磁盘读取或写入到磁盘之后是否被修改过。查看一个块是否需要写入时,我们只需要去查看 `uvpt` 条目中的 `PTE_D` 的 ”dirty“ 位即可。(`PTE_D` 位由处理器设置,用于表示那个页被写入;具体细节可以查看 x386 参考手册的 [第 5 章][3] 的 5.2.4.3 节)块被写入到磁盘上之后,`flush_block` 函数将使用 `sys_page_map` 去清除 `PTE_D` 位。
|
||||||
|
|
||||||
使用 `make grade` 去测试你的代码。你的代码应该能够通过 "check_bc"、"check_super"、和 "check_bitmap" 的测试。
|
> 使用 `make grade` 去测试你的代码。你的代码应该能够通过 "check_bc"、"check_super"、和 "check_bitmap" 的测试。
|
||||||
```
|
|
||||||
|
|
||||||
在 `fs/fs.c` 中的函数 `fs_init` 是块缓存使用的一个很好的示例。在初始化块缓存之后,它简单地在全局变量 `super` 中保存指针到磁盘映射区。在这之后,如果块在内存中,或我们的页故障服务程序按需将它们从磁盘上读入后,我们就能够简单地从 `super` 结构中读取块了。
|
在 `fs/fs.c` 中的函数 `fs_init` 是块缓存使用的一个很好的示例。在初始化块缓存之后,它简单地在全局变量 `super` 中保存指针到磁盘映射区。在这之后,如果块在内存中,或我们的页故障服务程序按需将它们从磁盘上读入后,我们就能够简单地从 `super` 结构中读取块了。
|
||||||
|
|
||||||
```markdown
|
.
|
||||||
小挑战!到现在为止,块缓存还没有清除策略。一旦某个块因为页故障被读入到缓存中之后,它将一直不会被清除,并且永远保留在内存中。给块缓存增加一个清除策略。在页表中使用 `PTE_A` 的 "accessed" 位来实现,任何环境访问一个页时,硬件就会设置这个位,你可以通过它来跟踪磁盘块的大致使用情况,而不需要修改访问磁盘映射区域的任何代码。使用脏块要小心。
|
|
||||||
```
|
> **小挑战!**到现在为止,块缓存还没有清除策略。一旦某个块因为页故障被读入到缓存中之后,它将一直不会被清除,并且永远保留在内存中。给块缓存增加一个清除策略。在页表中使用 `PTE_A` 的 "accessed" 位来实现,任何环境访问一个页时,硬件就会设置这个位,你可以通过它来跟踪磁盘块的大致使用情况,而不需要修改访问磁盘映射区域的任何代码。使用脏块要小心。
|
||||||
|
|
||||||
### 块位图
|
### 块位图
|
||||||
|
|
||||||
在 `fs_init` 设置了 `bitmap` 指针之后,我们可以认为 `bitmap` 是一个装满比特位的数组,磁盘上的每个块就是数组中的其中一个比特位。比如 `block_is_free`,它只是简单地在位图中检查给定的块是否被标记为空闲。
|
在 `fs_init` 设置了 `bitmap` 指针之后,我们可以认为 `bitmap` 是一个装满比特位的数组,磁盘上的每个块就是数组中的其中一个比特位。比如 `block_is_free`,它只是简单地在位图中检查给定的块是否被标记为空闲。
|
||||||
|
|
||||||
```markdown
|
> **练习 3**、使用 `free_block` 作为实现 `fs/fs.c` 中的 `alloc_block` 的一个模型,它将在位图中去查找一个空闲的磁盘块,并将它标记为已使用,然后返回块编号。当你分配一个块时,你应该立即使用 `flush_block` 将已改变的位图块刷新到磁盘上,以确保文件系统的一致性。
|
||||||
练习 3、使用 `free_block` 作为实现 `fs/fs.c` 中的 `alloc_block` 的一个模型,它将在位图中去查找一个空闲的磁盘块,并将它标记为已使用,然后返回块编号。当你分配一个块时,你应该立即使用 `flush_block` 将已改变的位图块刷新到磁盘上,以确保文件系统的一致性。
|
|
||||||
|
|
||||||
使用 `make grade` 去测试你的代码。现在,你的代码应该要通过 "alloc_block" 的测试。
|
> 使用 `make grade` 去测试你的代码。现在,你的代码应该要通过 "alloc_block" 的测试。
|
||||||
```
|
|
||||||
|
|
||||||
### 文件操作
|
### 文件操作
|
||||||
|
|
||||||
在 `fs/fs.c` 中,我们提供一系列的函数去实现基本的功能,比如,你将需要去理解和管理结构 `File`、扫描和管理目录”文件“的条目、 以及从根目录开始遍历文件系统以解析一个绝对路径名。阅读 `fs/fs.c` 中的所有代码,并在你开始实验之前,确保你理解了每个函数的功能。
|
在 `fs/fs.c` 中,我们提供一系列的函数去实现基本的功能,比如,你将需要去理解和管理结构 `File`、扫描和管理目录”文件“的条目、 以及从根目录开始遍历文件系统以解析一个绝对路径名。阅读 `fs/fs.c` 中的所有代码,并在你开始实验之前,确保你理解了每个函数的功能。
|
||||||
|
|
||||||
```markdown
|
> **练习 4**、实现 `file_block_walk` 和 `file_get_block`。`file_block_walk` 从一个文件中的块偏移量映射到 `struct File` 中那个块的指针上或间接块上,它非常类似于 `pgdir_walk` 在页表上所做的事。`file_get_block` 将更进一步,将去映射一个真实的磁盘块,如果需要的话,去分配一个新的磁盘块。
|
||||||
练习 4、实现 `file_block_walk` 和 `file_get_block`。`file_block_walk` 从一个文件中的块偏移量映射到 `struct File` 中那个块的指针上或间接块上,它非常类似于 `pgdir_walk` 在页表上所做的事。`file_get_block` 将更进一步,将去映射一个真实的磁盘块,如果需要的话,去分配一个新的磁盘块。
|
|
||||||
|
|
||||||
使用 `make grade` 去测试你的代码。你的代码应该要通过 "file_open"、"file_get_block"、以及 "file_flush/file_truncated/file rewrite"、和 "testfile" 的测试。
|
> 使用 `make grade` 去测试你的代码。你的代码应该要通过 "file_open"、"file_get_block"、以及 "file_flush/file_truncated/file rewrite"、和 "testfile" 的测试。
|
||||||
```
|
|
||||||
|
|
||||||
`file_block_walk` 和 `file_get_block` 是文件系统中的”劳动模范“。比如,`file_read` 和 `file_write` 或多或少都在 `file_get_block` 上做必需的登记工作,然后在分散的块和连续的缓存之间复制字节。
|
`file_block_walk` 和 `file_get_block` 是文件系统中的”劳动模范“。比如,`file_read` 和 `file_write` 或多或少都在 `file_get_block` 上做必需的登记工作,然后在分散的块和连续的缓存之间复制字节。
|
||||||
|
|
||||||
```
|
.
|
||||||
小挑战!如果操作在中途实然被打断(比如,突然崩溃或重启),文件系统很可能会产生错误。实现软件更新或日志处理的方式让文件系统的”崩溃可靠性“更好,并且演示一下旧的文件系统可能会崩溃,而你的更新后的文件系统不会崩溃的情况。
|
|
||||||
```
|
> **小挑战!**如果操作在中途实然被打断(比如,突然崩溃或重启),文件系统很可能会产生错误。实现软件更新或日志处理的方式让文件系统的”崩溃可靠性“更好,并且演示一下旧的文件系统可能会崩溃,而你的更新后的文件系统不会崩溃的情况。
|
||||||
|
|
||||||
### 文件系统接口
|
### 文件系统接口
|
||||||
|
|
||||||
@ -207,19 +197,17 @@ x86 处理器在 EFLAGS 寄存器中使用 IOPL 位去确定保护模式中的
|
|||||||
|
|
||||||
服务器也通过 IPC 来发送响应。我们为函数的返回代码使用 32 位的数字。对于大多数 RPC,这已经涵盖了它们全部的返回代码。`FSREQ_READ` 和 `FSREQ_STAT` 也返回数据,它们只是被简单地写入到客户端发送它的请求时的页上。在 IPC 的响应中并不需要去发送这个页,因为这个页是文件系统服务器和客户端从一开始就共享的页。另外,在它的响应中,`FSREQ_OPEN` 与客户端共享一个新的 "Fd page”。我们将快捷地返回到文件描述符页上。
|
服务器也通过 IPC 来发送响应。我们为函数的返回代码使用 32 位的数字。对于大多数 RPC,这已经涵盖了它们全部的返回代码。`FSREQ_READ` 和 `FSREQ_STAT` 也返回数据,它们只是被简单地写入到客户端发送它的请求时的页上。在 IPC 的响应中并不需要去发送这个页,因为这个页是文件系统服务器和客户端从一开始就共享的页。另外,在它的响应中,`FSREQ_OPEN` 与客户端共享一个新的 "Fd page”。我们将快捷地返回到文件描述符页上。
|
||||||
|
|
||||||
```markdown
|
> **练习 5**、实现 `fs/serv.c` 中的 `serve_read`。
|
||||||
练习 5、实现 `fs/serv.c` 中的 `serve_read`。
|
|
||||||
|
|
||||||
`serve_read` 的重任将由已经在 `fs/fs.c` 中实现的 `file_read` 来承担(它实际上不过是对 `file_get_block` 的一连串调用)。对于文件读取,`serve_read` 只能提供 RPC 接口。查看 `serve_set_size` 中的注释和代码,去大体上了解服务器函数的结构。
|
> `serve_read` 的重任将由已经在 `fs/fs.c` 中实现的 `file_read` 来承担(它实际上不过是对 `file_get_block` 的一连串调用)。对于文件读取,`serve_read` 只能提供 RPC 接口。查看 `serve_set_size` 中的注释和代码,去大体上了解服务器函数的结构。
|
||||||
|
|
||||||
使用 `make grade` 去测试你的代码。你的代码通过 "serve_open/file_stat/file_close" 和 "file_read" 的测试后,你得分应该是 70(总分为 150)。
|
> 使用 `make grade` 去测试你的代码。你的代码通过 "serve_open/file_stat/file_close" 和 "file_read" 的测试后,你得分应该是 70(总分为 150)。
|
||||||
```
|
|
||||||
|
|
||||||
```markdown
|
.
|
||||||
练习 6、实现 `fs/serv.c` 中的 `serve_write` 和 `lib/file.c` 中的 `devfile_write`。
|
|
||||||
|
|
||||||
使用 `make grade` 去测试你的代码。你的代码通过 "file_write"、"file_read after file_write"、"open"、和 "large file" 的测试后,得分应该是 90(总分为150)。
|
> **练习 6**、实现 `fs/serv.c` 中的 `serve_write` 和 `lib/file.c` 中的 `devfile_write`。
|
||||||
```
|
|
||||||
|
> 使用 `make grade` 去测试你的代码。你的代码通过 "file_write"、"file_read after file_write"、"open"、和 "large file" 的测试后,得分应该是 90(总分为150)。
|
||||||
|
|
||||||
### 进程增殖
|
### 进程增殖
|
||||||
|
|
||||||
@ -227,21 +215,19 @@ x86 处理器在 EFLAGS 寄存器中使用 IOPL 位去确定保护模式中的
|
|||||||
|
|
||||||
我们实现的是 `spawn`,而不是一个类 UNIX 的 `exec`,因为 `spawn` 是很容易从用户空间中、以”外内核式“ 实现的,它无需来自内核的特别帮助。为了在用户空间中实现 `exec`,想一想你应该做什么?确保你理解了它为什么很难。
|
我们实现的是 `spawn`,而不是一个类 UNIX 的 `exec`,因为 `spawn` 是很容易从用户空间中、以”外内核式“ 实现的,它无需来自内核的特别帮助。为了在用户空间中实现 `exec`,想一想你应该做什么?确保你理解了它为什么很难。
|
||||||
|
|
||||||
```markdown
|
> **练习 7**、`spawn` 依赖新的系统调用 `sys_env_set_trapframe` 去初始化新创建的环境的状态。实现 `kern/syscall.c` 中的 `sys_env_set_trapframe`。(不要忘记在 `syscall()` 中派发新系统调用)
|
||||||
练习 7、`spawn` 依赖新的系统调用 `sys_env_set_trapframe` 去初始化新创建的环境的状态。实现 `kern/syscall.c` 中的 `sys_env_set_trapframe`。(不要忘记在 `syscall()` 中派发新系统调用)
|
|
||||||
|
|
||||||
运行来自 `kern/init.c` 中的 `user/spawnhello` 程序来测试你的代码`kern/init.c` ,它将尝试从文件系统中增殖 `/hello`。
|
> 运行来自 `kern/init.c` 中的 `user/spawnhello` 程序来测试你的代码`kern/init.c` ,它将尝试从文件系统中增殖 `/hello`。
|
||||||
|
|
||||||
使用 `make grade` 去测试你的代码。
|
> 使用 `make grade` 去测试你的代码。
|
||||||
```
|
|
||||||
|
|
||||||
```markdown
|
.
|
||||||
小挑战!实现 Unix 式的 `exec`。
|
|
||||||
```
|
|
||||||
|
|
||||||
```markdown
|
> **小挑战!**实现 Unix 式的 `exec`。
|
||||||
小挑战!实现 `mmap` 式的文件内存映射,并如果可能的话,修改 `spawn` 从 ELF 中直接映射页。
|
|
||||||
```
|
.
|
||||||
|
|
||||||
|
> **小挑战!**实现 `mmap` 式的文件内存映射,并如果可能的话,修改 `spawn` 从 ELF 中直接映射页。
|
||||||
|
|
||||||
### 跨 fork 和 spawn 共享库状态
|
### 跨 fork 和 spawn 共享库状态
|
||||||
|
|
||||||
@ -255,11 +241,9 @@ UNIX 文件描述符是一个通称的概念,它还包括管道、控制台 I/
|
|||||||
|
|
||||||
我们在 `inc/lib.h` 中定义了一个新的 `PTE_SHARE` 位,在 Intel 和 AMD 的手册中,这个位是被标记为”软件可使用的“的三个 PTE 位之一。我们将创建一个约定,如果一个页表条目中这个位被设置,那么在 `fork` 和 `spawn` 中应该直接从父环境中复制 PTE 到子环境中。注意它与标记为写时复制的差别:正如在第一段中所描述的,我们希望确保能共享页更新。
|
我们在 `inc/lib.h` 中定义了一个新的 `PTE_SHARE` 位,在 Intel 和 AMD 的手册中,这个位是被标记为”软件可使用的“的三个 PTE 位之一。我们将创建一个约定,如果一个页表条目中这个位被设置,那么在 `fork` 和 `spawn` 中应该直接从父环境中复制 PTE 到子环境中。注意它与标记为写时复制的差别:正如在第一段中所描述的,我们希望确保能共享页更新。
|
||||||
|
|
||||||
```markdown
|
> **练习 8**、修改 `lib/fork.c` 中的 `duppage`,以遵循最新约定。如果页表条目设置了 `PTE_SHARE` 位,仅直接复制映射。(你应该去使用 `PTE_SYSCALL`,而不是 `0xfff`,去从页表条目中掩掉有关的位。`0xfff` 仅选出可访问的位和脏位。)
|
||||||
练习 8、修改 `lib/fork.c` 中的 `duppage`,以遵循最新约定。如果页表条目设置了 `PTE_SHARE` 位,仅直接复制映射。(你应该去使用 `PTE_SYSCALL`,而不是 `0xfff`,去从页表条目中掩掉有关的位。`0xfff` 仅选出可访问的位和脏位。)
|
|
||||||
|
|
||||||
同样的,在 `lib/spawn.c` 中实现 `copy_shared_pages`。它应该循环遍历当前进程中所有的页表条目(就像 `fork` 那样),复制任何设置了 `PTE_SHARE` 位的页映射到子进程中。
|
> 同样的,在 `lib/spawn.c` 中实现 `copy_shared_pages`。它应该循环遍历当前进程中所有的页表条目(就像 `fork` 那样),复制任何设置了 `PTE_SHARE` 位的页映射到子进程中。
|
||||||
```
|
|
||||||
|
|
||||||
使用 `make run-testpteshare` 去检查你的代码行为是否正确。正确的情况下,你应该会看到像 "`fork handles PTE_SHARE right`" 和 "`spawn handles PTE_SHARE right`” 这样的输出行。
|
使用 `make run-testpteshare` 去检查你的代码行为是否正确。正确的情况下,你应该会看到像 "`fork handles PTE_SHARE right`" 和 "`spawn handles PTE_SHARE right`” 这样的输出行。
|
||||||
|
|
||||||
@ -269,9 +253,7 @@ UNIX 文件描述符是一个通称的概念,它还包括管道、控制台 I/
|
|||||||
|
|
||||||
为了能让 shell 工作,我们需要一些方式去输入。QEMU 可以显示输出,我们将它的输出写入到 CGA 显示器上和串行端口上,但到目前为止,我们仅能够在内核监视器中接收输入。在 QEMU 中,我们从图形化窗口中的输入作为从键盘到 JOS 的输入,虽然键入到控制台的输入作为出现在串行端口上的字符的方式显现。在 `kern/console.c` 中已经包含了由我们自实验 1 以来的内核监视器所使用的键盘和串行端口的驱动程序,但现在你需要去把这些增加到系统中。
|
为了能让 shell 工作,我们需要一些方式去输入。QEMU 可以显示输出,我们将它的输出写入到 CGA 显示器上和串行端口上,但到目前为止,我们仅能够在内核监视器中接收输入。在 QEMU 中,我们从图形化窗口中的输入作为从键盘到 JOS 的输入,虽然键入到控制台的输入作为出现在串行端口上的字符的方式显现。在 `kern/console.c` 中已经包含了由我们自实验 1 以来的内核监视器所使用的键盘和串行端口的驱动程序,但现在你需要去把这些增加到系统中。
|
||||||
|
|
||||||
```markdown
|
> **练习 9**、在你的 `kern/trap.c` 中,调用 `kbd_intr` 去处理捕获 `IRQ_OFFSET+IRQ_KBD` 和 `serial_intr`,用它们去处理捕获 `IRQ_OFFSET+IRQ_SERIAL`。
|
||||||
练习 9、在你的 `kern/trap.c` 中,调用 `kbd_intr` 去处理捕获 `IRQ_OFFSET+IRQ_KBD` 和 `serial_intr`,用它们去处理捕获 `IRQ_OFFSET+IRQ_SERIAL`。
|
|
||||||
```
|
|
||||||
|
|
||||||
在 `lib/console.c` 中,我们为你实现了文件的控制台输入/输出。`kbd_intr` 和 `serial_intr` 将使用从最新读取到的输入来填充缓冲区,而控制台文件类型去排空缓冲区(默认情况下,控制台文件类型为 stdin/stdout,除非用户重定向它们)。
|
在 `lib/console.c` 中,我们为你实现了文件的控制台输入/输出。`kbd_intr` 和 `serial_intr` 将使用从最新读取到的输入来填充缓冲区,而控制台文件类型去排空缓冲区(默认情况下,控制台文件类型为 stdin/stdout,除非用户重定向它们)。
|
||||||
|
|
||||||
@ -282,40 +264,38 @@ UNIX 文件描述符是一个通称的概念,它还包括管道、控制台 I/
|
|||||||
运行 `make run-icode` 或 `make run-icode-nox` 将运行你的内核并启动 `user/icode`。`icode` 又运行 `init`,它将设置控制台作为文件描述符 0 和 1(即:标准输入 stdin 和标准输出 stdout),然后增殖出环境 `sh`,就是 shell。之后你应该能够运行如下的命令了:
|
运行 `make run-icode` 或 `make run-icode-nox` 将运行你的内核并启动 `user/icode`。`icode` 又运行 `init`,它将设置控制台作为文件描述符 0 和 1(即:标准输入 stdin 和标准输出 stdout),然后增殖出环境 `sh`,就是 shell。之后你应该能够运行如下的命令了:
|
||||||
|
|
||||||
```
|
```
|
||||||
echo hello world | cat
|
echo hello world | cat
|
||||||
cat lorem |cat
|
cat lorem |cat
|
||||||
cat lorem |num
|
cat lorem |num
|
||||||
cat lorem |num |num |num |num |num
|
cat lorem |num |num |num |num |num
|
||||||
lsfd
|
lsfd
|
||||||
```
|
```
|
||||||
|
|
||||||
注意用户库常规程序 `cprintf` 将直接输出到控制台,而不会使用文件描述符代码。这对调试非常有用,但是对管道连接其它程序却很不利。为将输出打印到特定的文件描述符(比如 1,它是标准输出 stdout),需要使用 `fprintf(1, "...", ...)`。`printf("...", ...)` 是将输出打印到文件描述符 1(标准输出 stdout) 的快捷方式。查看 `user/lsfd.c` 了解更多示例。
|
注意用户库常规程序 `cprintf` 将直接输出到控制台,而不会使用文件描述符代码。这对调试非常有用,但是对管道连接其它程序却很不利。为将输出打印到特定的文件描述符(比如 1,它是标准输出 stdout),需要使用 `fprintf(1, "...", ...)`。`printf("...", ...)` 是将输出打印到文件描述符 1(标准输出 stdout) 的快捷方式。查看 `user/lsfd.c` 了解更多示例。
|
||||||
|
|
||||||
```markdown
|
> **练习 10**、这个 shell 不支持 I/O 重定向。如果能够运行 `run sh <script` 就更完美了,就不用将所有的命令手动去放入一个脚本中,就像上面那样。为 `<` 在 `user/sh.c` 中添加重定向的功能。
|
||||||
练习 10、
|
|
||||||
这个 shell 不支持 I/O 重定向。如果能够运行 `run sh <script` 就更完美了,就不用将所有的命令手动去放入一个脚本中,就像上面那样。为 `<` 在 `user/sh.c` 中添加重定向的功能。
|
|
||||||
|
|
||||||
通过在你的 shell 中输入 `sh <script` 来测试你实现的重定向功能。
|
> 通过在你的 shell 中输入 `sh <script` 来测试你实现的重定向功能。
|
||||||
|
|
||||||
运行 `make run-testshell` 去测试你的 shell。`testshell` 只是简单地给 shell ”喂“上面的命令(也可以在 `fs/testshell.sh` 中找到),然后检查它的输出是否与 `fs/testshell.key` 一样。
|
> 运行 `make run-testshell` 去测试你的 shell。`testshell` 只是简单地给 shell ”喂“上面的命令(也可以在 `fs/testshell.sh` 中找到),然后检查它的输出是否与 `fs/testshell.key` 一样。
|
||||||
```
|
|
||||||
|
|
||||||
```markdown
|
.
|
||||||
小挑战!给你的 shell 添加更多的特性。最好包括以下的特性(其中一些可能会要求修改文件系统):
|
|
||||||
|
|
||||||
* 后台命令 (`ls &`)
|
|
||||||
* 一行中运行多个命令 (`ls; echo hi`)
|
|
||||||
* 命令组 (`(ls; echo hi) | cat > out`)
|
|
||||||
* 扩展环境变量 (`echo $hello`)
|
|
||||||
* 引号 (`echo "a | b"`)
|
|
||||||
* 命令行历史和/或编辑功能
|
|
||||||
* tab 命令补全
|
|
||||||
* 为命令行查找目录、cd 和路径
|
|
||||||
* 文件创建
|
|
||||||
* 用快捷键 `ctl-c` 去杀死一个运行中的环境
|
|
||||||
|
|
||||||
可做的事情还有很多,并不仅限于以上列表。
|
> **小挑战!**给你的 shell 添加更多的特性。最好包括以下的特性(其中一些可能会要求修改文件系统):
|
||||||
```
|
|
||||||
|
> * 后台命令 (`ls &`)
|
||||||
|
> * 一行中运行多个命令 (`ls; echo hi`)
|
||||||
|
> * 命令组 (`(ls; echo hi) | cat > out`)
|
||||||
|
> * 扩展环境变量 (`echo $hello`)
|
||||||
|
> * 引号 (`echo "a | b"`)
|
||||||
|
> * 命令行历史和/或编辑功能
|
||||||
|
> * tab 命令补全
|
||||||
|
> * 为命令行查找目录、cd 和路径
|
||||||
|
> * 文件创建
|
||||||
|
> * 用快捷键 `ctl-c` 去杀死一个运行中的环境
|
||||||
|
|
||||||
|
> 可做的事情还有很多,并不仅限于以上列表。
|
||||||
|
|
||||||
到目前为止,你的代码应该要通过所有的测试。和以前一样,你可以使用 `make grade` 去评级你的提交,并且使用 `make handin` 上交你的实验。
|
到目前为止,你的代码应该要通过所有的测试。和以前一样,你可以使用 `make grade` 去评级你的提交,并且使用 `make handin` 上交你的实验。
|
||||||
|
|
||||||
@ -328,7 +308,7 @@ via: https://pdos.csail.mit.edu/6.828/2018/labs/lab5/
|
|||||||
作者:[csail.mit][a]
|
作者:[csail.mit][a]
|
||||||
选题:[lujun9972][b]
|
选题:[lujun9972][b]
|
||||||
译者:[qhwdw](https://github.com/qhwdw)
|
译者:[qhwdw](https://github.com/qhwdw)
|
||||||
校对:[校对者ID](https://github.com/校对者ID)
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
@ -0,0 +1,180 @@
|
|||||||
|
通过询问-响应身份认证提高桌面登录安全
|
||||||
|
======
|
||||||
|
|
||||||
|
![](https://fedoramagazine.org/wp-content/uploads/2018/10/challenge-response-816x345.png)
|
||||||
|
|
||||||
|
### 介绍
|
||||||
|
|
||||||
|
今天,Fedora 提供了多种方式来提高我们账户的身份认证的安全性。当然,它有我们熟悉的用户名密码登录方式,它也同样提供了其他的身份认证选项,比如生物识别、指纹、智能卡、一次性密码,甚至是<ruby>询问-响应<rt>challenge-response</rt></ruby>身份认证。
|
||||||
|
|
||||||
|
每种认证方式都有明确的优缺点。这点本身就可以成为一篇相当冗长的文章的主题。Fedora 杂志之前就已经介绍过了这其中的一些选项:
|
||||||
|
|
||||||
|
- [在 Fedora 中使用 YubiKey4][1]
|
||||||
|
- [Fedora 28:在 OpenSSH 中更好的支持智能卡][2]
|
||||||
|
|
||||||
|
在现在的 Fedora 版本中,最安全的方法之一就是离线硬件询问-响应。它也同样是最容易部署的方法之一。下面是具体方法。
|
||||||
|
|
||||||
|
### 询问-响应认证
|
||||||
|
|
||||||
|
从技术上来讲,当你输入密码的时候,你就正在响应用户名询问。离线的询问、响应包含了这些部分:首先是需要你的用户名,接下来,Fedora 会要你提供一个加密的物理硬件的令牌。令牌会把另一个其存储的加密密钥通过<ruby>可插入式身份认证<rt>Pluggable Authentication Module</rt></ruby>模块(PAM)框架来响应询问。最后,Fedora 才会提示你输入密码。这可以防止其他人仅仅使用了找到的硬件令牌,或是只使用了账户名密码而没有正确的加密密钥。
|
||||||
|
|
||||||
|
这意味着除了你的账户名密码之外,你必须事先在你的操作系统中注册了一个或多个加密硬件令牌。你必须保证你的物理硬件令牌能够匹配你的用户名。
|
||||||
|
|
||||||
|
一些询问-响应的方法,比如一次性密码(OTP),在硬件令牌上获取加密的代码密钥,然后将这个密钥通过网络传输到远程身份认证服务器。然后这个服务器会告诉 Fedora 的 PAM 框架,这是否是该用户的一个有效令牌。如果身份认证服务器在本地网络上,这个方法非常好。但它的缺点是如果网络连接断开或是你在没有网的远程端工作。你会被锁在系统之外,直到你能通过网络连接到身份认证服务器。
|
||||||
|
|
||||||
|
有时候,生产环境会采用通过 Yubikey 使用一次性密码(OTP)的设置,然而,在家庭或个人的系统上,你可能更喜欢询问-响应设置。一切都是本地的,这种方法不需要通过远程网络调用。下面这些过程适用于 Fedora 27、28 和 29.
|
||||||
|
|
||||||
|
### 准备
|
||||||
|
|
||||||
|
#### 硬件令牌密钥
|
||||||
|
|
||||||
|
首先,你需要一个安全的硬件令牌密钥。具体来说,这个过程需要一个 Yubikey 4、Yubikey NEO,或者是最近发布的、同样支持 FIDO2 的 Yubikey 5 系列设备。你应该购买它们中的两个,一个做备份,以避免其中一个丢失或遭到损坏。你可以在不同的工作地点使用这些密钥。较为简单的 FIDO 和 FIDO U2F 版本不适用于这个过程,但是非常适合使用 FIDO 的在线服务。
|
||||||
|
|
||||||
|
#### 备份、备份,以及备份
|
||||||
|
|
||||||
|
接下来,为你所有的重要数据制作备份,你可能想在克隆在 VM 里的 Fedora 27/28/29 里测试配置,来确保你在设置你自己的个人工作环境之前理解这个过程。
|
||||||
|
|
||||||
|
#### 升级,然后安装
|
||||||
|
|
||||||
|
现在,确定你的 Fedora 是最新的,然后通过 `dnf` 命令安装所需要的 Fedora Yubikey 包。
|
||||||
|
|
||||||
|
```
|
||||||
|
$ sudo dnf upgrade
|
||||||
|
$ sudo dnf install ykclient* ykpers* pam_yubico*
|
||||||
|
```
|
||||||
|
|
||||||
|
如果你使用的是 VM 环境,例如 Virtual Box,确保 Yubikey 设备已经插进了 USB 口,然后允许 VM 控制的 USB 访问 Yubikey。
|
||||||
|
|
||||||
|
### 配置 Yubikey
|
||||||
|
|
||||||
|
确认你的账户访问到了 USB Yubikey:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ ykinfo -v
|
||||||
|
version: 3.5.0
|
||||||
|
```
|
||||||
|
|
||||||
|
如果 Yubikey 没有被检测到,会出现下面这些错误信息:
|
||||||
|
|
||||||
|
```
|
||||||
|
Yubikey core error: no yubikey present
|
||||||
|
```
|
||||||
|
|
||||||
|
接下来,通过下面这些 `ykpersonalize` 命令初始化你每个新的 Yubikey。这将设置 Yubikey 配置插槽 2 使用 HMAC-SHA1 算法(即使少于 64 个字符)进行询问响应。如果你已经为询问响应设置好了你的 Yubikey。你就不需要再次运行 `ykpersonalize` 了。
|
||||||
|
|
||||||
|
```
|
||||||
|
ykpersonalize -2 -ochal-resp -ochal-hmac -ohmac-lt64 -oserial-api-visible
|
||||||
|
```
|
||||||
|
|
||||||
|
一些用户在使用的时候将 YubiKey 留在了他们的工作站上,甚至用于对虚拟机进行询问-响应。然而,为了更好的安全性,你可能会更愿意使用手动触发 YubiKey 来响应询问。
|
||||||
|
|
||||||
|
要添加手动询问按钮触发器,请添加 `-ochal-btn-trig` 选项,这个选项可以使得 Yubikey 在请求中闪烁其 LED。等待你在 15 秒内按下硬件密钥区域上的按钮来生成响应密钥。
|
||||||
|
|
||||||
|
```
|
||||||
|
$ ykpersonalize -2 -ochal-resp -ochal-hmac -ohmac-lt64 -ochal-btn-trig -oserial-api-visible
|
||||||
|
```
|
||||||
|
|
||||||
|
为你的每个新的硬件密钥执行此操作。每个密钥执行一次。完成编程之后,使用下面的命令将 Yubikey 配置存储到 `~/.yubico`:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ ykpamcfg -2 -v
|
||||||
|
debug: util.c:222 (check_firmware_version): YubiKey Firmware version: 4.3.4
|
||||||
|
|
||||||
|
Sending 63 bytes HMAC challenge to slot 2
|
||||||
|
Sending 63 bytes HMAC challenge to slot 2
|
||||||
|
Stored initial challenge and expected response in '/home/chuckfinley/.yubico/challenge-9992567'.
|
||||||
|
```
|
||||||
|
|
||||||
|
如果你要设置多个密钥用于备份,请将所有的密钥设置为相同,然后使用 `ykpamcfg` 工具存储每个密钥的询问-响应。如果你在一个已经存在的注册密钥上运行 `ykpersonalize` 命令,你就必须再次存储配置信息。
|
||||||
|
|
||||||
|
### 配置 /etc/pam.d/sudo
|
||||||
|
|
||||||
|
现在要去验证配置是否有效,**在同一个终端窗口中**,你需要设置 `sudo` 来要求使用 Yubikey 的询问-响应。将下面这几行插入到 `/etc/pam.d/sudo` 文件中。
|
||||||
|
|
||||||
|
```
|
||||||
|
auth required pam_yubico.so mode=challenge-response
|
||||||
|
```
|
||||||
|
|
||||||
|
将上面的 `auth` 行插入到文件中的 `auth include system-auth` 行的上面,然后保存并退出编辑器。在默认的 Fedora 29 设置中,`/etc/pam.d/sudo` 应该像下面这样:
|
||||||
|
|
||||||
|
```
|
||||||
|
#%PAM-1.0
|
||||||
|
auth required pam_yubico.so mode=challenge-response
|
||||||
|
auth include system-auth
|
||||||
|
account include system-auth
|
||||||
|
password include system-auth
|
||||||
|
session optional pam_keyinit.so revoke
|
||||||
|
session required pam_limits.so
|
||||||
|
session include system-auth
|
||||||
|
```
|
||||||
|
|
||||||
|
**保持这个初始的终端窗口打开**,然后打开一个新的终端窗口进行测试,在新的终端窗口中输入:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ sudo echo testing
|
||||||
|
```
|
||||||
|
|
||||||
|
你应该注意到了 Yubikey 上的 LED 在闪烁。点击 Yubikey 按钮,你应该会看见一个输入 `sudo` 密码的提示。在你输入你的密码之后,你应该会在终端屏幕上看见 “testing” 的字样。
|
||||||
|
|
||||||
|
现在去测试确保失败也正常,启动另一个终端窗口,并从 USB 插口中拔掉 Yubikey。使用下面这条命令验证,在没有 Yubikey 的情况下,`sudo` 是否会不再正常工作。
|
||||||
|
|
||||||
|
```
|
||||||
|
$ sudo echo testing fail
|
||||||
|
```
|
||||||
|
|
||||||
|
你应该立刻被提示输入 `sudo` 密码,但即使你输入了正确密码,登录也应该失败。
|
||||||
|
|
||||||
|
### 设置 Gnome 桌面管理器(GDM)
|
||||||
|
|
||||||
|
一旦你的测试完成后,你就可以为图形登录添加询问-响应支持了。将你的 Yubikey 再次插入进 USB 插口中。然后将下面这几行添加到 `/etc/pam.d/gdm-password` 文件中:
|
||||||
|
|
||||||
|
```
|
||||||
|
auth required pam_yubico.so mode=challenge-response
|
||||||
|
```
|
||||||
|
|
||||||
|
打开一个终端窗口,然后运行下面这些命令。如果需要,你可以使用其他的编辑器:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ sudo vi /etc/pam.d/gdm-password
|
||||||
|
```
|
||||||
|
|
||||||
|
你应该看到 Yubikey 上的 LED 在闪烁,按下 Yubikey 按钮,然后在提示符处输入密码。
|
||||||
|
|
||||||
|
修改 `/etc/pam.d/gdm-password` 文件,在已有的 `auth substack password-auth` 行上添加新的行。这个文件的顶部应该像下面这样:
|
||||||
|
|
||||||
|
```
|
||||||
|
auth [success=done ignore=ignore default=bad] pam_selinux_permit.so
|
||||||
|
auth required pam_yubico.so mode=challenge-response
|
||||||
|
auth substack password-auth
|
||||||
|
auth optional pam_gnome_keyring.so
|
||||||
|
auth include postlogin
|
||||||
|
|
||||||
|
account required pam_nologin.so
|
||||||
|
```
|
||||||
|
|
||||||
|
保存更改并退出编辑器,如果你使用的是 vi,输入键是按 `Esc` 键,然后在提示符处输入 `wq!` 来保存并退出。
|
||||||
|
|
||||||
|
### 结论
|
||||||
|
|
||||||
|
现在注销 GNOME。将 Yubikey 插入到 USB 口,在图形登录界面上点击你的用户名。Yubikey LED 会开始闪烁。触摸那个按钮,你会被提示输入你的密码。
|
||||||
|
|
||||||
|
如果你丢失了 Yubikey,除了重置密码之外,你还可以使用备份的 Yubikey。你还可以给你的账户增加额外的 Yubikey 配置。
|
||||||
|
|
||||||
|
如果有其他人获得了你的密码,他们在没有你的物理硬件 Yubikey 的情况下,仍然不能登录。恭喜!你已经显著提高了你的工作环境登录的安全性了。
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: https://fedoramagazine.org/login-challenge-response-authentication/
|
||||||
|
|
||||||
|
作者:[nabooengineer][a]
|
||||||
|
选题:[lujun9972][b]
|
||||||
|
译者:[hopefully2333](https://github.com/hopefully2333)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]: https://fedoramagazine.org/author/nabooengineer/
|
||||||
|
[b]: https://github.com/lujun9972
|
||||||
|
[1]: https://fedoramagazine.org/using-the-yubikey4-with-fedora/
|
||||||
|
[2]: https://fedoramagazine.org/fedora-28-better-smart-card-support-openssh/
|
||||||
|
|
@ -1,31 +1,33 @@
|
|||||||
持续集成与部署的3个最佳实践
|
持续集成与部署的 3 个最佳实践
|
||||||
======
|
======
|
||||||
了解自动化,使用 Git 存储库以及参数化 Jenkins 管道。
|
|
||||||
|
> 了解自动化,使用 Git 存储库以及参数化 Jenkins 管道。
|
||||||
|
|
||||||
![](https://opensource.com/sites/default/files/styles/image-full-size/public/lead-images/innovation_lightbulb_gears_devops_ansible.png?itok=TSbmp3_M)
|
![](https://opensource.com/sites/default/files/styles/image-full-size/public/lead-images/innovation_lightbulb_gears_devops_ansible.png?itok=TSbmp3_M)
|
||||||
|
|
||||||
本文涵盖了三个关键主题:自动化 CI/CD 配置、使用 Git 存储库处理常见的 CI/CD 工件、参数化 Jenkins 管道。
|
本文涵盖了三个关键主题:自动化 CI/CD 配置、使用 Git 存储库处理常见的 CI/CD 工件、参数化 Jenkins 管道。
|
||||||
|
|
||||||
### 术语
|
### 术语
|
||||||
|
|
||||||
首先,我们定义一些术语。**CI/CD** 是允许团队快速自动化测试、打包、部署其应用程序的实践。它通常通过利用名为 **[Jenkins][1]** 的服务器来实现,该服务器充当 CI/CD 协调器。Jenkins 侦听特定输入(通常是代码签入后的 Git hook),并在触发时启动管道。
|
首先,我们定义一些术语。**CI/CD** 是允许团队快速自动化测试、打包、部署其应用程序的实践。它通常通过利用名为 [Jenkins][1] 的服务器来实现,该服务器充当 CI/CD 协调器。Jenkins 侦听特定输入(通常是代码签入后的 Git 挂钩),并在触发时启动一个管道。
|
||||||
|
|
||||||
**pipeline** 由开发和/或运营团队编写的代码组成,这些代码指导 Jenkins 在 CI/CD 过程中采取哪些操作。这个流水线通常类似于“构建我的代码,然后测试我的代码,如果这些测试通过,则把我的应用程序部署到下一个最高环境(通常是开发、测试或生产环境)”。组织通常具有更复杂的流水线,并入了诸如工件存储库和代码分析器之类的工具,但是这提供了一个高级示例。
|
<ruby>管道<rt>pipeline</rt></ruby> 由开发和/或运营团队编写的代码组成,这些代码指导 Jenkins 在 CI/CD 过程中采取哪些操作。这个流水线通常类似于“构建我的代码,然后测试我的代码,如果这些测试通过,则把我的应用程序部署到下一个最高环境(通常是开发、测试或生产环境)”。组织通常具有更复杂的管道,并入了诸如工件存储库和代码分析器之类的工具,这里提供了一个高级示例。
|
||||||
|
|
||||||
现在我们了解了关键术语,让我们深入研究一些最佳实践。
|
现在我们了解了关键术语,让我们深入研究一些最佳实践。
|
||||||
|
|
||||||
### 1\. 自动化是关键
|
### 1、自动化是关键
|
||||||
|
|
||||||
要在 PaaS 上运行 CI/CD,需要在集群上配置适当的基础设施。在这个例子中,我将使用 [OpenShift][2]。
|
要在 PaaS 上运行 CI/CD,需要在集群上配置适当的基础设施。在这个例子中,我将使用 [OpenShift][2]。
|
||||||
|
|
||||||
"Hello, World" 的实现很容易实现。简单地运行 **oc new-app jenkins- <persistent/ephemeral>** 和 voilà, 你已经准备好运行 Jenkins 服务器了。然而,在企业中的使用要复杂得多。除了 Jenkins 服务器之外,管理员通常还需要部署代码分析工具(如 SonarQube)和构件库(如 Nexus)。然后,它们必须创建管道来执行 CI/CD 和 Jenkins 从服务器,以减少主服务器的负载。这些实体中的大多数都由 OpenShift 资源支持,需要创建这些资源来部署所需的 CI/CD 基础设施。
|
“Hello, World” 的实现很容易实现。简单地运行 `oc new-app jenkins-<persistent/ephemeral>`,然后,你就有了一个已经就绪的运行中的 Jenkins 服务器了。然而,在企业中的使用要复杂得多。除了 Jenkins 服务器之外,管理员通常还需要部署代码分析工具(如 SonarQube)和工件库(如 Nexus)。然后,它们必须创建管道来执行 CI/CD 和 Jenkins 从服务器,以减少主服务器的负载。这些实体中的大多数都由 OpenShift 资源支持,需要创建这些资源来部署所需的 CI/CD 基础设施。
|
||||||
|
|
||||||
最后,部署 CI/CD 组件所需要的手动步骤可能是需要被重复的,并且你可能不想成为执行那些重复步骤的人。为了确保结果能够像以前一样快速、无错误和准确地产生,应该在创建基础设施的方式中结合自动化方法。这可以是一个 Ansible 剧本、一个 Bash 脚本,或者任何您希望自动化 CI/CD 基础设施部署的其他方式。我已经使用 [Ansible][3] 和 [OpenShift-Applier][4] 角色来自动化我的实现。您可能会发现这些工具很有价值,或者您可能会发现其他一些对您和组织更有效的工具。无论哪种方式,您都将发现自动化显著地减少了重新创建 CI/CD 组件所需的工作量。
|
最后,部署 CI/CD 组件所需要的手动步骤可能是需要重复进行的,而且你可能不想成为执行那些重复步骤的人。为了确保结果能够像以前一样快速、无错误和准确地产生,应该在创建基础设施的方式中结合自动化方法。这可以是一个 Ansible 剧本、一个 Bash 脚本,或者任何您希望自动化 CI/CD 基础设施部署的其它方式。我已经使用 [Ansible][3] 和 [OpenShift-Applier][4] 角色来自动化我的实现。您可能会发现这些工具很有价值,或者您可能会发现其他一些对您和组织更有效的工具。无论哪种方式,您都将发现自动化显著地减少了重新创建 CI/CD 组件所需的工作量。
|
||||||
|
|
||||||
#### 配置Jenkins主服务器
|
#### 配置 Jenkins 主服务器
|
||||||
|
|
||||||
除了一般的“自动化”之外,我想单独介绍一下 Jenkins 主服务器,并讨论管理员如何利用 OpenShift 来自动化配置 Jenkins。来自 [Red Hat Container Catalog][5] 的 Jenkins 映像已经安装了 [OpenShift-Sync plugin][6]。在 [视频][7] 中,我们将讨论如何使用这个插件来创建 Jenkins 管道和从设备。
|
除了一般的“自动化”之外,我想单独介绍一下 Jenkins 主服务器,并讨论管理员如何利用 OpenShift 来自动化配置 Jenkins。来自 [Red Hat Container Catalog][5] 的 Jenkins 镜像已经安装了 [OpenShift-Sync plugin][6]。在 [该视频][7] 中,我们将讨论如何使用这个插件来创建 Jenkins 管道和从设备。
|
||||||
|
|
||||||
要创建 Jenkins 流水线,请创建一个类似于下面的 OpenShift BuildConfig:
|
要创建 Jenkins 管道,请创建一个类似于下面的 OpenShift BuildConfig:
|
||||||
|
|
||||||
```
|
```
|
||||||
apiVersion: v1
|
apiVersion: v1
|
||||||
@ -43,7 +45,7 @@ spec:
|
|||||||
type: JenkinsPipeline
|
type: JenkinsPipeline
|
||||||
```
|
```
|
||||||
|
|
||||||
OpenShift-Sync 插件将注意到已经创建了带有 **jenkinsPipelineStrategy** 策略的 BuildConfig,并将其转换为 Jenkins 管道,从 Git 源指定的 Jenkins 文件中提取。也可以使用内联 Jenkinsfile,而不是从 Git 存储库中提取。有关更多信息,请参阅[文档][8]。
|
OpenShift-Sync 插件将注意到已经创建了带有 `jenkinsPipelineStrategy` 策略的 BuildConfig,并将其转换为 Jenkins 管道,从 Git 源指定的 Jenkinsfile 中提取。也可以使用内联 Jenkinsfile,而不是从 Git 存储库中提取。有关更多信息,请参阅[文档][8]。
|
||||||
|
|
||||||
要创建 Jenkins 从站,请创建一个 OpenShift ImageStream,它从以下定义开始:
|
要创建 Jenkins 从站,请创建一个 OpenShift ImageStream,它从以下定义开始:
|
||||||
|
|
||||||
@ -55,12 +57,12 @@ metadata:
|
|||||||
slave-label: jenkins-slave
|
slave-label: jenkins-slave
|
||||||
labels:
|
labels:
|
||||||
role: jenkins-slave
|
role: jenkins-slave
|
||||||
…
|
...
|
||||||
```
|
```
|
||||||
|
|
||||||
注意在这个 ImageStream 中定义的元数据。OpenShift-Sync 插件将把带有标签 **role: jenkins-slave** 的任何ImageStream 转换为 Jenkins 从站。Jenkins 从站将以 **slave-label** 注释中的值命名。
|
注意在这个 ImageStream 中定义的元数据。OpenShift-Sync 插件将把带有标签 `role: jenkins-slave` 的任何 ImageStream 转换为 Jenkins 从站。Jenkins 从站将以 `slave-label` 注释中的值命名。
|
||||||
|
|
||||||
ImageStreams 对于简单的 Jenkins 从属配置工作得很好,但是一些团队会发现有必要配置一些细节详情,比如资源限制、准备就绪和活动性探测,以及实例帽。这就是 ConfigMap 发挥作用的地方:
|
ImageStreams 对于简单的 Jenkins 从属配置工作得很好,但是一些团队会发现有必要配置一些细节详情,比如资源限制、准备就绪和活动性探测,以及实例上限。这就是 ConfigMap 发挥作用的地方:
|
||||||
|
|
||||||
```
|
```
|
||||||
apiVersion: v1
|
apiVersion: v1
|
||||||
@ -74,21 +76,21 @@ data:
|
|||||||
<Kubernetes pod template>
|
<Kubernetes pod template>
|
||||||
```
|
```
|
||||||
|
|
||||||
注意,仍然需要角色:jenkins-slave 标签来将 ConfigMap 转换为 Jenkins slave。Kubernetes pod 模板由一长段 XML 组成,它将根据组织的喜好配置每个细节。要查看此 XML,以及有关将 ImageStreams 和 ConfigMaps 转换为 Jenkins 从属的更多信息,请参阅[文档][9]。
|
注意,仍然需要角色:`jenkins-slave` 标签来将 ConfigMap 转换为 Jenkins 从站。Kubernetes pod 模板由一长段 XML 组成,它将根据组织的喜好配置每个细节。要查看此 XML,以及有关将 ImageStreams 和 ConfigMaps 转换为 Jenkins 从站的更多信息,请参阅[文档][9]。
|
||||||
|
|
||||||
请注意上面所示的三个示例,其中没有一个操作需要管理员对 Jenkins 控制台进行手动更改。通过使用 OpenShift 资源,可以简单的自动化方式配置 Jenkins。
|
请注意上面所示的三个示例,其中没有一个操作需要管理员对 Jenkins 控制台进行手动更改。通过使用 OpenShift 资源,可以简单的自动化方式配置 Jenkins。
|
||||||
|
|
||||||
### 2\. 分享就是关心
|
### 2、分享就是关爱
|
||||||
|
|
||||||
第二个最佳实践是维护一个公共 CI/CD 工件的 Git 存储库。主要思想是防止团队重新发明轮子。假设您的团队需要执行到 OpenShift 环境的蓝色/绿色部署,作为管道 CD 阶段的一部分。负责编写流水线的团队成员可能不是 OpenShift 专家,也不可能具有从头开始编写此功能的能力。幸运的是,有人已经编写了一个将此功能合并到一个公共 CI/CD 存储库中的函数,因此您的团队可以使用该函数而不是花时间编写一个函数。
|
第二个最佳实践是维护一个公共 CI/CD 工件的 Git 存储库。主要思想是防止团队重新发明轮子。假设您的团队需要执行到 OpenShift 环境的蓝/绿部署,作为管道 CD 阶段的一部分。负责编写管道的团队成员可能不是 OpenShift 专家,也不可能具有从头开始编写此功能的能力。幸运的是,有人已经编写了一个将此功能合并到一个公共 CI/CD 存储库中的函数,因此您的团队可以使用该函数而不是花时间编写一个函数。
|
||||||
|
|
||||||
为了更进一步,您的组织可能决定维护整个管道。您可能会发现团队正在编写具有相似功能的流水线。对于那些团队来说,使用来自公共存储库的参数化管道要比从头开始编写自己的管道更有效。
|
为了更进一步,您的组织可能决定维护整个管道。您可能会发现团队正在编写具有相似功能的管道。对于那些团队来说,使用来自公共存储库的参数化管道要比从头开始编写自己的管道更有效。
|
||||||
|
|
||||||
### 3\. 少即是多
|
### 3、少即是多
|
||||||
|
|
||||||
正如我在前一节中提到的,第三个也是最后一个最佳实践是参数化您的 CI/CD 管道。参数化将防止过多的管道,使您的 CI/CD 系统更容易维护。假设我有多个区域可以部署应用程序。如果没有参数化,我需要为每个区域设置单独的管道。
|
正如我在前一节中提到的,第三个也是最后一个最佳实践是参数化您的 CI/CD 管道。参数化将防止过多的管道,使您的 CI/CD 系统更容易维护。假设我有多个区域可以部署应用程序。如果没有参数化,我需要为每个区域设置单独的管道。
|
||||||
|
|
||||||
要参数化作为 OpenShift 构建配置编写的管道,请将 **env** 节添加到配置:
|
要参数化一个作为 OpenShift 构建配置编写的管道,请将 `env` 节添加到配置:
|
||||||
|
|
||||||
```
|
```
|
||||||
...
|
...
|
||||||
@ -103,9 +105,9 @@ spec:
|
|||||||
type: JenkinsPipeline
|
type: JenkinsPipeline
|
||||||
```
|
```
|
||||||
|
|
||||||
使用此配置,我可以传递 **REGION** 参数管道以将我的应用程序部署到指定区域。
|
使用此配置,我可以传递 `REGION` 参数给管道以将我的应用程序部署到指定区域。
|
||||||
|
|
||||||
这有一个[视频][7]提供了一个更实质性的情况,其中参数化是必须的。一些组织决定把他们的 CI/CD 管道分割成单独的 CI 和 CD 管道,通常是因为在部署之前有一些审批过程。假设我有四个映像和三个不同的环境要部署。如果没有参数化,我需要12个 CD 管道来允许所有部署可能性。这会很快失去控制。为了使 CD 流水线的维护更容易,组织会发现将映像和环境参数化以便允许一个流水线执行多个流水线的工作会更好。
|
这有一个[视频][7]提供了一个更实质性的情况,其中参数化是必须的。一些组织决定把他们的 CI/CD 管道分割成单独的 CI 和 CD 管道,通常是因为在部署之前有一些审批过程。假设我有四个镜像和三个不同的环境要部署。如果没有参数化,我需要 12 个 CD 管道来允许所有部署可能性。这会很快失去控制。为了使 CD 流水线的维护更容易,组织会发现将镜像和环境参数化以便允许一个流水线执行多个流水线的工作会更好。
|
||||||
|
|
||||||
### 总结
|
### 总结
|
||||||
|
|
||||||
@ -120,7 +122,7 @@ via: https://opensource.com/article/18/11/best-practices-cicd
|
|||||||
作者:[Austin Dewey][a]
|
作者:[Austin Dewey][a]
|
||||||
选题:[lujun9972][b]
|
选题:[lujun9972][b]
|
||||||
译者:[ChiZelin](https://github.com/ChiZelin)
|
译者:[ChiZelin](https://github.com/ChiZelin)
|
||||||
校对:[校对者ID](https://github.com/校对者ID)
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
73
published/201812/20181121 DevOps is for everyone.md
Normal file
73
published/201812/20181121 DevOps is for everyone.md
Normal file
@ -0,0 +1,73 @@
|
|||||||
|
[#]: collector: (lujun9972)
|
||||||
|
[#]: translator: (alim0x)
|
||||||
|
[#]: reviewer: (wxy)
|
||||||
|
[#]: publisher: (wxy)
|
||||||
|
[#]: subject: (DevOps is for everyone)
|
||||||
|
[#]: via: (https://opensource.com/article/18/11/how-non-engineer-got-devops)
|
||||||
|
[#]: author: (Dawn Parych https://opensource.com/users/dawnparzych)
|
||||||
|
[#]: url: (https://linux.cn/article-10386-1.html)
|
||||||
|
|
||||||
|
所有人的 DevOps
|
||||||
|
======
|
||||||
|
|
||||||
|
> 让一名非工程师来解释为什么你不必成为一位开发者或运维就能爱上 DevOps。
|
||||||
|
|
||||||
|
![](https://opensource.com/sites/default/files/styles/image-full-size/public/lead-images/team-game-play-inclusive-diversity-collaboration.png?itok=8sUXV7W1)
|
||||||
|
|
||||||
|
我从未做过开发或运维的工作 —— 那怎么我在写一篇关于 [DevOps][1] 的文章?我一直都对计算机和技术有兴趣。我还对社群、心理学以及帮助他人充满热情。当我第一次听到 DevOps 时,这个概念激起了我的兴趣,因为它看起来融合了很多我感兴趣的东西,即便我是不写代码的。
|
||||||
|
|
||||||
|
我的第一台电脑是 TRS-80,我喜欢在上面编写 BASIC 程序。我只上过两门我的高中开设的计算机编程课程。若干年后,我创办了一家计算机公司。我定制邮件标签和信纸,并建立了一个数据库来存储地址。
|
||||||
|
|
||||||
|
问题是我并不能从写代码中获得享受。我想要教育和帮助人们,我没法将写代码看作这样的一个机会。是的,技术可以帮助人们并改变生活,但是写代码没有点燃我的热情。我需要对我的工作感到兴奋并做我喜欢的事情。
|
||||||
|
|
||||||
|
我发现我爱 DevOps。对我而言,DevOps 指的是:
|
||||||
|
|
||||||
|
* 文化,而不是代码
|
||||||
|
* 过程,而不是结果
|
||||||
|
* 建立一个所有人可以持续提升的环境
|
||||||
|
* 沟通与合作,而不是独立工作
|
||||||
|
|
||||||
|
归根结底,DevOps 是指成为社区工作的一部分,实现共同的目标。DevOps 融合了心理学、社群、技术。DevOps 不是一个职位名称,它是一种生活和工作的哲学。
|
||||||
|
|
||||||
|
### 找到我的社群
|
||||||
|
|
||||||
|
快四年前,我在西雅图参加了我的第一个 [DevOps 日][2] 会议。我感觉我找到了我的社群。我觉得受到了欢迎和接受,尽管我从事营销工作而且没有计算机科学文凭。我可以从心理学和技术中寻找乐趣。
|
||||||
|
|
||||||
|
在 DevOps 日,我学到了 [DevOps“三步工作法”][3] —— 流动、反馈、持续实验和学习 —— 以及新(对我而言)的概念,如 Kaizen(改善)和 Kaikaku(改革)。随着我的学习深入,我发现我在说这样的话,“我是这样做的!我都不知道这样做还有个名字!”
|
||||||
|
|
||||||
|
[Kaizen(改善)][4]是持续改进和学习的实践。小的量变积累随着时间的推移可以引起质变。我发现它和卡罗尔·德韦克的[成长型思维][5]的想法很相似。人们不是生来就是专家。在某方面拥有经验需要花费时间、练习,以及常常还有失败。承认增量的改善对确保我们不会放弃是很有必要的。
|
||||||
|
|
||||||
|
另一方面,[Kaikaku(改革)][6]的概念是指,长时间的小的改变有时不能起作用,你需要做一些完全的或破坏性的改变。在没有找到下份工作前就辞职或移居新城市就足够有破坏性 —— 是的,两者我都做过。但这些彻底的改变收获巨大。如果我没有辞职并休息一段时间,我也许不会接触到 DevOps。等我决定继续工作的时候,我一直听到 DevOps,我开始研究它。这引导我参加了我的第一个 DevOps 日,从那里我开始看到我的所有热情开始聚集。从那时起,我已经参加了五次 DevOps 日活动,并且定期撰写关于 DevOps 话题的文章。
|
||||||
|
|
||||||
|
### 将三步工作法用到工作中
|
||||||
|
|
||||||
|
改变是困难的,学习新事物可以听起来很吓人。DevOps 的三步工作法提供了一个管理改变的框架。比如:信息流动是怎样的?是什么驱动着你做出改变?一旦你认为一个改变是必需的,你如何获得这个改变是否正确的反馈?你如何知道你在取得进展?反馈是必要的,并且应该包含积极和有建设性的要素。困难的地方在于保证建设性的要素不要重于积极要素。
|
||||||
|
|
||||||
|
对我而言,第三步 —— 持续实验和学习 —— 是 DevOps 最重要的部分。有一个可以自由地实验和冒险的环境,人们可以获得意想不到的结果。有时这些结果是好的,有时不是太好 —— 但这没事。创建一个可以接受失败结果的环境可以鼓励人们冒险。我们都应该力争定期的持续实验和学习。
|
||||||
|
|
||||||
|
DevOps 的三步工作法提供了一个尝试,获得反馈,以及从错误中获取经验的方法。几年前,我的儿子告诉我,“我从来就没想做到最好,因为那样我就没法从我的错误中学到东西了。”我们都会犯错,从中获得经验帮助我们成长和改善。如果我们的文化不支持尝试和学习,我们就不会愿意去犯错。
|
||||||
|
|
||||||
|
### 成为社区的一部分
|
||||||
|
|
||||||
|
我已经在技术领域工作了超过 20 年,直到我发现 DevOps 社区前,我还经常感觉自己是个外行。如果你像我一样——对技术充满热情,但不是工程和运维那方面——你仍然可以成为 DevOps 的一部分,即便你从事的是销售、营销、产品营销、技术写作、支持或其他工作。DevOps 是属于所有人的。
|
||||||
|
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: https://opensource.com/article/18/11/how-non-engineer-got-devops
|
||||||
|
|
||||||
|
作者:[Dawn Parych][a]
|
||||||
|
选题:[lujun9972][b]
|
||||||
|
译者:[alim0x](https://github.com/alim0x)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]: https://opensource.com/users/dawnparzych
|
||||||
|
[b]: https://github.com/lujun9972
|
||||||
|
[1]: https://opensource.com/resources/devops
|
||||||
|
[2]: https://www.devopsdays.org/
|
||||||
|
[3]: https://itrevolution.com/the-three-ways-principles-underpinning-devops/
|
||||||
|
[4]: https://en.wikipedia.org/wiki/Kaizen
|
||||||
|
[5]: https://en.wikipedia.org/wiki/Mindset#Fixed_and_growth
|
||||||
|
[6]: https://en.wikipedia.org/wiki/Kaikaku
|
@ -1,47 +1,51 @@
|
|||||||
[#]: collector: (lujun9972)
|
[#]: collector: (lujun9972)
|
||||||
[#]: translator: (qhwdw)
|
[#]: translator: (qhwdw)
|
||||||
[#]: reviewer:
|
[#]: reviewer: (wxy)
|
||||||
[#]: publisher:
|
[#]: publisher: (wxy)
|
||||||
[#]: subject: (How to Build a Netboot Server, Part 1)
|
[#]: subject: (How to Build a Netboot Server, Part 1)
|
||||||
|
|
||||||
[#]: via: (https://fedoramagazine.org/how-to-build-a-netboot-server-part-1/)
|
[#]: via: (https://fedoramagazine.org/how-to-build-a-netboot-server-part-1/)
|
||||||
[#]: author: (Gregory Bartholomew https://fedoramagazine.org/author/glb/)
|
[#]: author: (Gregory Bartholomew https://fedoramagazine.org/author/glb/)
|
||||||
|
[#]: url: (https://linux.cn/article-10379-1.html)
|
||||||
|
|
||||||
[#]: url:
|
如何构建一台网络引导服务器(一)
|
||||||
|
======
|
||||||
如何构建一台网络引导服务器(第一部分)
|
|
||||||
======
|
|
||||||
|
|
||||||
![](https://fedoramagazine.org/wp-content/uploads/2018/11/build-netboot-816x345.jpg)
|
![](https://fedoramagazine.org/wp-content/uploads/2018/11/build-netboot-816x345.jpg)
|
||||||
|
|
||||||
有些计算机网络需要在各个物理机器上维护相同的软件和配置。学校的计算机实验室就是这样的一个环境。一台 [网络引导][1] 服务器能够被配置为基于网络去提供一个完整的操作系统,以便于客户端计算机从一个中央位置获取配置。本教程将向你展示构建一台网络引导服务器的一种方法。
|
有些计算机网络需要在各个物理机器上维护相同的软件和配置。学校的计算机实验室就是这样的一个环境。 [网络引导][1] 服务器能够被配置为基于网络去提供一个完整的操作系统,以便于客户端计算机从一个中央位置获取配置。本教程将向你展示构建一台网络引导服务器的一种方法。
|
||||||
|
|
||||||
本教程的第一部分将包括创建一台网络引导服务器和镜像。第二部分将展示如何去添加 Kerberos 验证的 home 目录到网络引导配置中。
|
本教程的第一部分将包括创建一台网络引导服务器和镜像。第二部分将展示如何去添加 Kerberos 验证的家目录到网络引导配置中。
|
||||||
|
|
||||||
### 初始化配置
|
### 初始化配置
|
||||||
|
|
||||||
首先去下载 Fedora 服务器的 [netinst][2] 镜像,将它刻录到一张光盘上,然后它将引导服务器去重新格式化。我们只需要一个典型的 Fedora Server 的“最小化安装”来作为我们的开端,安装完成后,我们可以使用命令行去添加我们需要的任何额外的包。
|
首先去下载 Fedora 服务器的 [netinst][2] 镜像,将它刻录到一张光盘上,然后用它引导服务器来重新格式化。我们只需要一个典型的 Fedora Server 的“最小化安装”来作为我们的开端,安装完成后,我们可以使用命令行去添加我们需要的任何额外的包。
|
||||||
|
|
||||||
![][3]
|
![][3]
|
||||||
|
|
||||||
> 注意:本教程中我们将使用 Fedora 28。其它版本在“最小化安装”中包含的包可能略有不同。如果你使用的是不同的 Fedora 版本,如果一个预期的文件或命令不可用,你可能需要做一些调试。
|
> 注意:本教程中我们将使用 Fedora 28。其它版本在“最小化安装”中包含的包可能略有不同。如果你使用的是不同的 Fedora 版本,如果一个预期的文件或命令不可用,你可能需要做一些调试。
|
||||||
|
|
||||||
最小化安装的 Fedora Server 运行起来之后,以 root 用户登入并设置主机名字:
|
最小化安装的 Fedora Server 运行起来之后,以 root 用户登入:
|
||||||
|
|
||||||
```javascript
|
```
|
||||||
|
$ sudo -i
|
||||||
|
```
|
||||||
|
|
||||||
|
并设置主机名字:
|
||||||
|
|
||||||
|
```
|
||||||
$ MY_HOSTNAME=server-01.example.edu
|
$ MY_HOSTNAME=server-01.example.edu
|
||||||
$ hostnamectl set-hostname $MY_HOSTNAME
|
$ hostnamectl set-hostname $MY_HOSTNAME
|
||||||
```
|
```
|
||||||
|
|
||||||
> 注意:Red Hat 建议静态和临时名字应都要与这个机器在 DNS 中的完全合格域名相匹配,比如 host.example.com([了解主机名字][4])。
|
> 注意:Red Hat 建议静态和临时名字应都要与这个机器在 DNS 中的完全合格域名相匹配,比如 host.example.com([了解主机名字][4])。
|
||||||
>
|
>
|
||||||
> 注意:本指南为了你“复制粘贴”友好。需要自定义的任何值都声明为一个 MY_* 变量,在你运行剩余命令之前,你可能需要调整它。如果你注销之后,变量的赋值将被清除。
|
> 注意:本指南为了方便“复制粘贴”。需要自定义的任何值都声明为一个 `MY_*` 变量,在你运行剩余命令之前,你可能需要调整它。如果你注销之后,变量的赋值将被清除。
|
||||||
>
|
>
|
||||||
> 注意:Fedora 28 Server 在默认情况下往往会转储大量的日志到控制台上。你可以通过运行命令:sysctl -w kernel.printk=0 去禁用控制台日志输出。
|
> 注意:Fedora 28 Server 在默认情况下往往会转储大量的日志到控制台上。你可以通过运行命令:`sysctl -w kernel.printk=0` 去禁用控制台日志输出。
|
||||||
|
|
||||||
接下来,我们需要在我们的服务器上配置一个静态网络地址。运行下面的一系列命令将找到并重新配置你的默认网络连接:
|
接下来,我们需要在我们的服务器上配置一个静态网络地址。运行下面的一系列命令将找到并重新配置你的默认网络连接:
|
||||||
|
|
||||||
```javascript
|
```
|
||||||
$ MY_DNS1=192.0.2.91
|
$ MY_DNS1=192.0.2.91
|
||||||
$ MY_DNS2=192.0.2.92
|
$ MY_DNS2=192.0.2.92
|
||||||
$ MY_IP=192.0.2.158
|
$ MY_IP=192.0.2.158
|
||||||
@ -66,7 +70,7 @@ nmcli con up br0-slave0
|
|||||||
END
|
END
|
||||||
```
|
```
|
||||||
|
|
||||||
> 注意:上面最后的一组命令被封装到一个 “nohup” 脚本中,因为它将临时禁用网络。这个 nohup 命令将允许 nmcli 命令去完成运行,直到你的 SSH 连接断开。注意,连接恢复可能需要 10 秒左右的时间,如果你改变了服务器 IP 地址,你将需要重新启动一个新的 SSH 连接。
|
> 注意:上面最后的一组命令被封装到一个 `nohup` 脚本中,因为它将临时禁用网络。这个 `nohup` 命令可以让 `nmcli` 命令运行完成,即使你的 SSH 连接断开。注意,连接恢复可能需要 10 秒左右的时间,如果你改变了服务器 IP 地址,你将需要重新启动一个新的 SSH 连接。
|
||||||
>
|
>
|
||||||
> 注意:上面的网络配置在默认的连接之上创建了一个 [网桥][5],这样我们在后面的测试中就可以直接运行一个虚拟机实例。如果你不想在这台服务器上去直接测试网络引导镜像,你可以跳过创建网桥的命令,并直接在你的默认网络连接上配置静态 IP 地址。
|
> 注意:上面的网络配置在默认的连接之上创建了一个 [网桥][5],这样我们在后面的测试中就可以直接运行一个虚拟机实例。如果你不想在这台服务器上去直接测试网络引导镜像,你可以跳过创建网桥的命令,并直接在你的默认网络连接上配置静态 IP 地址。
|
||||||
|
|
||||||
@ -80,26 +84,26 @@ $ dnf install -y nfs-utils
|
|||||||
|
|
||||||
为发布 NFS 去创建一个顶级的 [伪文件系统][6],然后在你的网络上共享它:
|
为发布 NFS 去创建一个顶级的 [伪文件系统][6],然后在你的网络上共享它:
|
||||||
|
|
||||||
```javascript
|
```
|
||||||
$ MY_SUBNET=192.0.2.0
|
$ MY_SUBNET=192.0.2.0
|
||||||
$ mkdir /export
|
$ mkdir /export
|
||||||
$ echo "/export -fsid=0,ro,sec=sys,root_squash $MY_SUBNET/$MY_PREFIX" > /etc/exports
|
$ echo "/export -fsid=0,ro,sec=sys,root_squash $MY_SUBNET/$MY_PREFIX" > /etc/exports
|
||||||
```
|
```
|
||||||
|
|
||||||
SELinux 将干扰网络引导服务器的运行。在本教程中我们将不涉及为它配置例外的部分,因此我们直接禁用它:
|
SELinux 会干扰网络引导服务器的运行。为它配置例外规则超出了本教程中,因此我们这里直接禁用它:
|
||||||
|
|
||||||
```javascript
|
```
|
||||||
$ sed -i '/GRUB_CMDLINE_LINUX/s/"$/ audit=0 selinux=0"/' /etc/default/grub
|
$ sed -i '/GRUB_CMDLINE_LINUX/s/"$/ audit=0 selinux=0"/' /etc/default/grub
|
||||||
$ grub2-mkconfig -o /boot/grub2/grub.cfg
|
$ grub2-mkconfig -o /boot/grub2/grub.cfg
|
||||||
$ sed -i 's/SELINUX=enforcing/SELINUX=disabled/' /etc/sysconfig/selinux
|
$ sed -i 's/SELINUX=enforcing/SELINUX=disabled/' /etc/sysconfig/selinux
|
||||||
$ setenforce 0
|
$ setenforce 0
|
||||||
```
|
```
|
||||||
|
|
||||||
> 注意:编辑 grub 命令行应该是不需要的,但在测试过程中发现,简单地编辑 /etc/sysconfig/selinux 被证明重启后是无效的,因此再次确保设置了 “selinux=0” 标志。
|
> 注意:应该不需要编辑 grub 命令行,但我们在测试过程中发现,直接编辑 `/etc/sysconfig/selinux` 被证明重启后是无效的,因此这样做再次确保设置了 `selinux=0` 标志。
|
||||||
|
|
||||||
现在,在本地防火墙中为 NFS 服务添加一个例外,然后启动 NFS 服务:
|
现在,在本地防火墙中为 NFS 服务添加一个例外规则,然后启动 NFS 服务:
|
||||||
|
|
||||||
```javascript
|
```
|
||||||
$ firewall-cmd --add-service nfs
|
$ firewall-cmd --add-service nfs
|
||||||
$ firewall-cmd --runtime-to-permanent
|
$ firewall-cmd --runtime-to-permanent
|
||||||
$ systemctl enable nfs-server.service
|
$ systemctl enable nfs-server.service
|
||||||
@ -116,76 +120,70 @@ $ systemctl start nfs-server.service
|
|||||||
$ mkdir /fc28
|
$ mkdir /fc28
|
||||||
```
|
```
|
||||||
|
|
||||||
使用 “dnf” 命令在新目录下用几个基础包去构建镜像:
|
使用 `dnf` 命令在新目录下用几个基础包去构建镜像:
|
||||||
|
|
||||||
```javascript
|
```
|
||||||
$ dnf -y --releasever=28 --installroot=/fc28 install fedora-release systemd passwd rootfiles sudo dracut dracut-network nfs-utils vim-minimal dnf
|
$ dnf -y --releasever=28 --installroot=/fc28 install fedora-release systemd passwd rootfiles sudo dracut dracut-network nfs-utils vim-minimal dnf
|
||||||
```
|
```
|
||||||
|
|
||||||
在上面的命令中省略了很重要的 “kernel” 包。在它们被安装完成之前,我们需要去调整一下 “initramfs” 镜像中包含的驱动程序集,“kernel” 首次安装时将自动构建这个镜像。尤其是,我们需要禁用 “hostonly” 模式,以便于 initramfs 镜像能够在各种硬件平台上正常工作,并且我们还需要添加对网络和 NFS 的支持:
|
在上面的命令中省略了很重要的 `kernel` 包。在它们被安装完成之前,我们需要去调整一下 `initramfs` 镜像中包含的驱动程序集,`kernel` 首次安装时将自动构建这个镜像。尤其是,我们需要禁用 `hostonly` 模式,以便于 `initramfs` 镜像能够在各种硬件平台上正常工作,并且我们还需要添加对网络和 NFS 的支持:
|
||||||
|
|
||||||
```javascript
|
```
|
||||||
$ echo 'hostonly=no' > /fc28/etc/dracut.conf.d/hostonly.conf
|
$ echo 'hostonly=no' > /fc28/etc/dracut.conf.d/hostonly.conf
|
||||||
$ echo 'add_dracutmodules+=" network nfs "' > /fc28/etc/dracut.conf.d/netboot.conf
|
$ echo 'add_dracutmodules+=" network nfs "' > /fc28/etc/dracut.conf.d/netboot.conf
|
||||||
```
|
```
|
||||||
|
|
||||||
现在,安装 kernel:
|
现在,安装 `kernel` 包:
|
||||||
|
|
||||||
```javascript
|
```
|
||||||
$ dnf -y --installroot=/fc28 install kernel
|
$ dnf -y --installroot=/fc28 install kernel
|
||||||
```
|
```
|
||||||
|
|
||||||
设置一个阻止 kernel 被更新的规则:
|
设置一个阻止 `kernel` 包被更新的规则:
|
||||||
|
|
||||||
```javascript
|
```
|
||||||
$ echo 'exclude=kernel-*' >> /fc28/etc/dnf/dnf.conf
|
$ echo 'exclude=kernel-*' >> /fc28/etc/dnf/dnf.conf
|
||||||
```
|
```
|
||||||
|
|
||||||
设置 locale:
|
设置 locale:
|
||||||
|
|
||||||
```javascript
|
```
|
||||||
$ echo 'LANG="en_US.UTF-8"' > /fc28/etc/locale.conf
|
$ echo 'LANG="en_US.UTF-8"' > /fc28/etc/locale.conf
|
||||||
```
|
```
|
||||||
|
|
||||||
> 注意:如果 locale 没有正确配置,一些程序(如 GNOME Terminal)将无法正常工作。
|
> 注意:如果 locale 没有正确配置,一些程序(如 GNOME Terminal)将无法正常工作。
|
||||||
|
|
||||||
root 用户密码留空:
|
|
||||||
|
|
||||||
```javascript
|
|
||||||
$ sed -i 's/^root:\*/root:/' /fc28/etc/shadow
|
|
||||||
```
|
|
||||||
|
|
||||||
设置客户端的主机名字:
|
设置客户端的主机名字:
|
||||||
|
|
||||||
```javascript
|
```
|
||||||
$ MY_CLIENT_HOSTNAME=client-01.example.edu
|
$ MY_CLIENT_HOSTNAME=client-01.example.edu
|
||||||
$ echo $MY_CLIENT_HOSTNAME > /fc28/etc/hostname
|
$ echo $MY_CLIENT_HOSTNAME > /fc28/etc/hostname
|
||||||
```
|
```
|
||||||
|
|
||||||
禁用控制台日志输出:
|
禁用控制台日志输出:
|
||||||
|
|
||||||
```javascript
|
```
|
||||||
$ echo 'kernel.printk = 0 4 1 7' > /fc28/etc/sysctl.d/00-printk.conf
|
$ echo 'kernel.printk = 0 4 1 7' > /fc28/etc/sysctl.d/00-printk.conf
|
||||||
```
|
```
|
||||||
|
|
||||||
定义网络引导镜像中的本地 “liveuser” 用户:
|
定义网络引导镜像中的本地 `liveuser` 用户:
|
||||||
|
|
||||||
```javascript
|
```
|
||||||
$ echo 'liveuser:x:1000:1000::/home/liveuser:/bin/bash' >> /fc28/etc/passwd
|
$ echo 'liveuser:x:1000:1000::/home/liveuser:/bin/bash' >> /fc28/etc/passwd
|
||||||
$ echo 'liveuser::::::::' >> /fc28/etc/shadow
|
$ echo 'liveuser::::::::' >> /fc28/etc/shadow
|
||||||
$ echo 'liveuser:x:1000:' >> /fc28/etc/group
|
$ echo 'liveuser:x:1000:' >> /fc28/etc/group
|
||||||
$ echo 'liveuser:!::' >> /fc28/etc/gshadow
|
$ echo 'liveuser:!::' >> /fc28/etc/gshadow
|
||||||
```
|
```
|
||||||
|
|
||||||
在 sudo 中启用 “liveuser”:
|
允许 `liveuser` 使用 `sudo`:
|
||||||
|
|
||||||
```javascript
|
```
|
||||||
$ echo 'liveuser ALL=(ALL) NOPASSWD: ALL' > /fc28/etc/sudoers.d/liveuser
|
$ echo 'liveuser ALL=(ALL) NOPASSWD: ALL' > /fc28/etc/sudoers.d/liveuser
|
||||||
```
|
```
|
||||||
|
|
||||||
启用自动 home 目录创建:
|
启用自动创建家目录:
|
||||||
|
|
||||||
```livescript
|
```
|
||||||
$ dnf install -y --installroot=/fc28 authselect oddjob-mkhomedir
|
$ dnf install -y --installroot=/fc28 authselect oddjob-mkhomedir
|
||||||
$ echo 'dirs /home' > /fc28/etc/rwtab.d/home
|
$ echo 'dirs /home' > /fc28/etc/rwtab.d/home
|
||||||
$ chroot /fc28 authselect select sssd with-mkhomedir --force
|
$ chroot /fc28 authselect select sssd with-mkhomedir --force
|
||||||
@ -194,19 +192,19 @@ $ chroot /fc28 systemctl enable oddjobd.service
|
|||||||
|
|
||||||
由于多个客户端将会同时挂载我们的镜像,我们需要去配置镜像工作在只读模式中:
|
由于多个客户端将会同时挂载我们的镜像,我们需要去配置镜像工作在只读模式中:
|
||||||
|
|
||||||
```livescript
|
```
|
||||||
$ sed -i 's/^READONLY=no$/READONLY=yes/' /fc28/etc/sysconfig/readonly-root
|
$ sed -i 's/^READONLY=no$/READONLY=yes/' /fc28/etc/sysconfig/readonly-root
|
||||||
```
|
```
|
||||||
|
|
||||||
配置日志输出到内存而不是持久存储中:
|
配置日志输出到内存而不是持久存储中:
|
||||||
|
|
||||||
```livescript
|
```
|
||||||
$ sed -i 's/^#Storage=auto$/Storage=volatile/' /fc28/etc/systemd/journald.conf
|
$ sed -i 's/^#Storage=auto$/Storage=volatile/' /fc28/etc/systemd/journald.conf
|
||||||
```
|
```
|
||||||
|
|
||||||
配置 DNS:
|
配置 DNS:
|
||||||
|
|
||||||
```livescript
|
```
|
||||||
$ MY_DNS1=192.0.2.91
|
$ MY_DNS1=192.0.2.91
|
||||||
$ MY_DNS2=192.0.2.92
|
$ MY_DNS2=192.0.2.92
|
||||||
$ cat << END > /fc28/etc/resolv.conf
|
$ cat << END > /fc28/etc/resolv.conf
|
||||||
@ -215,9 +213,9 @@ nameserver $MY_DNS2
|
|||||||
END
|
END
|
||||||
```
|
```
|
||||||
|
|
||||||
解决编写本教程时存在的只读 root 挂载 bug([BZ1542567][7]):
|
绕开编写本教程时存在的根目录只读挂载的 bug([BZ1542567][7]):
|
||||||
|
|
||||||
```livescript
|
```
|
||||||
$ echo 'dirs /var/lib/gssproxy' > /fc28/etc/rwtab.d/gssproxy
|
$ echo 'dirs /var/lib/gssproxy' > /fc28/etc/rwtab.d/gssproxy
|
||||||
$ cat << END > /fc28/etc/rwtab.d/systemd
|
$ cat << END > /fc28/etc/rwtab.d/systemd
|
||||||
dirs /var/lib/systemd/catalog
|
dirs /var/lib/systemd/catalog
|
||||||
@ -227,7 +225,7 @@ END
|
|||||||
|
|
||||||
最后,为我们镜像创建 NFS 文件系统,并将它共享到我们的子网中:
|
最后,为我们镜像创建 NFS 文件系统,并将它共享到我们的子网中:
|
||||||
|
|
||||||
```livescript
|
```
|
||||||
$ mkdir /export/fc28
|
$ mkdir /export/fc28
|
||||||
$ echo '/fc28 /export/fc28 none bind 0 0' >> /etc/fstab
|
$ echo '/fc28 /export/fc28 none bind 0 0' >> /etc/fstab
|
||||||
$ mount /export/fc28
|
$ mount /export/fc28
|
||||||
@ -237,20 +235,20 @@ $ exportfs -vr
|
|||||||
|
|
||||||
### 创建引导加载器
|
### 创建引导加载器
|
||||||
|
|
||||||
现在,我们已经有了可以进行网络引导的操作系统,我们需要一个引导加载器去从客户端系统上启动它。在本教程中我们使用的是 [iPXE][8].
|
现在,我们已经有了可以进行网络引导的操作系统,我们需要一个引导加载器去从客户端系统上启动它。在本教程中我们使用的是 [iPXE][8]。
|
||||||
|
|
||||||
> 注意:本节和接下来的节 — 使用 QEMU 测试 — 能在另外一台单独的计算机上来完成;它们不需要在网络引导服务器上来运行。
|
> 注意:本节和接下来的节使用 QEMU 测试,也能在另外一台单独的计算机上来完成;它们并不需要在网络引导服务器上来运行。
|
||||||
|
|
||||||
安装 git 并使用它去下载 iPXE:
|
安装 `git` 并使用它去下载 iPXE:
|
||||||
|
|
||||||
```livescript
|
```
|
||||||
$ dnf install -y git
|
$ dnf install -y git
|
||||||
$ git clone http://git.ipxe.org/ipxe.git $HOME/ipxe
|
$ git clone http://git.ipxe.org/ipxe.git $HOME/ipxe
|
||||||
```
|
```
|
||||||
|
|
||||||
现在我们需要去为我们的引导加载器创建一个指定的启动脚本:
|
现在我们需要去为我们的引导加载器创建一个指定的启动脚本:
|
||||||
|
|
||||||
```livescript
|
```
|
||||||
$ cat << 'END' > $HOME/ipxe/init.ipxe
|
$ cat << 'END' > $HOME/ipxe/init.ipxe
|
||||||
#!ipxe
|
#!ipxe
|
||||||
|
|
||||||
@ -264,19 +262,19 @@ END
|
|||||||
|
|
||||||
启动 “file” 下载协议:
|
启动 “file” 下载协议:
|
||||||
|
|
||||||
```livescript
|
```
|
||||||
$ echo '#define DOWNLOAD_PROTO_FILE' > $HOME/ipxe/src/config/local/general.h
|
$ echo '#define DOWNLOAD_PROTO_FILE' > $HOME/ipxe/src/config/local/general.h
|
||||||
```
|
```
|
||||||
|
|
||||||
安装 C 编译器以及相关的工具和库:
|
安装 C 编译器以及相关的工具和库:
|
||||||
|
|
||||||
```livescript
|
```
|
||||||
$ dnf groupinstall -y "C Development Tools and Libraries"
|
$ dnf groupinstall -y "C Development Tools and Libraries"
|
||||||
```
|
```
|
||||||
|
|
||||||
构建引导加载器:
|
构建引导加载器:
|
||||||
|
|
||||||
```livescript
|
```
|
||||||
$ cd $HOME/ipxe/src
|
$ cd $HOME/ipxe/src
|
||||||
$ make clean
|
$ make clean
|
||||||
$ make bin-x86_64-efi/ipxe.efi EMBED=../init.ipxe
|
$ make bin-x86_64-efi/ipxe.efi EMBED=../init.ipxe
|
||||||
@ -284,7 +282,7 @@ $ make bin-x86_64-efi/ipxe.efi EMBED=../init.ipxe
|
|||||||
|
|
||||||
记下新编译的引导加载器的存储位置。我们将在接下来的节中用到它:
|
记下新编译的引导加载器的存储位置。我们将在接下来的节中用到它:
|
||||||
|
|
||||||
```livescript
|
```
|
||||||
$ IPXE_FILE="$HOME/ipxe/src/bin-x86_64-efi/ipxe.efi"
|
$ IPXE_FILE="$HOME/ipxe/src/bin-x86_64-efi/ipxe.efi"
|
||||||
```
|
```
|
||||||
|
|
||||||
@ -292,13 +290,13 @@ $ IPXE_FILE="$HOME/ipxe/src/bin-x86_64-efi/ipxe.efi"
|
|||||||
|
|
||||||
这一节是可选的,但是你需要去复制下面显示在物理机器上的 [EFI 系统分区][9] 的布局,在网络引导时需要去配置它们。
|
这一节是可选的,但是你需要去复制下面显示在物理机器上的 [EFI 系统分区][9] 的布局,在网络引导时需要去配置它们。
|
||||||
|
|
||||||
> 注意:如果你想实现一个完全的无盘系统,你也可以复制那个文件到一个 TFTP 服务器,然后从 DHCP 上引用那台服务器。
|
> 注意:如果你想实现一个完全的无盘系统,你也可以复制那个文件到一个 TFTP 服务器,然后从 DHCP 上指向那台服务器。
|
||||||
|
|
||||||
为了使用 QEMU 去测试我们的引导加载器,我们继续去创建一个仅包含一个 EFI 系统分区和我们的启动文件的、很小的磁盘镜像。
|
为了使用 QEMU 去测试我们的引导加载器,我们继续去创建一个仅包含一个 EFI 系统分区和我们的启动文件的、很小的磁盘镜像。
|
||||||
|
|
||||||
从创建 EFI 系统分区所需要的目录布局开始,然后把我们在前面节中创建的引导加载器复制进去:
|
从创建 EFI 系统分区所需要的目录布局开始,然后把我们在前面节中创建的引导加载器复制进去:
|
||||||
|
|
||||||
```livescript
|
```
|
||||||
$ mkdir -p $HOME/esp/efi/boot
|
$ mkdir -p $HOME/esp/efi/boot
|
||||||
$ mkdir $HOME/esp/linux
|
$ mkdir $HOME/esp/linux
|
||||||
$ cp $IPXE_FILE $HOME/esp/efi/boot/bootx64.efi
|
$ cp $IPXE_FILE $HOME/esp/efi/boot/bootx64.efi
|
||||||
@ -306,13 +304,13 @@ $ cp $IPXE_FILE $HOME/esp/efi/boot/bootx64.efi
|
|||||||
|
|
||||||
下面的命令将识别我们的引导加载器镜像正在使用的内核版本,并将它保存到一个变量中,以备后续的配置命令去使用它:
|
下面的命令将识别我们的引导加载器镜像正在使用的内核版本,并将它保存到一个变量中,以备后续的配置命令去使用它:
|
||||||
|
|
||||||
```livescript
|
```
|
||||||
$ DEFAULT_VER=$(ls -c /fc28/lib/modules | head -n 1)
|
$ DEFAULT_VER=$(ls -c /fc28/lib/modules | head -n 1)
|
||||||
```
|
```
|
||||||
|
|
||||||
定义我们的客户端计算机将使用的引导配置:
|
定义我们的客户端计算机将使用的引导配置:
|
||||||
|
|
||||||
```livescript
|
```
|
||||||
$ MY_DNS1=192.0.2.91
|
$ MY_DNS1=192.0.2.91
|
||||||
$ MY_DNS2=192.0.2.92
|
$ MY_DNS2=192.0.2.92
|
||||||
$ MY_NFS4=server-01.example.edu
|
$ MY_NFS4=server-01.example.edu
|
||||||
@ -329,14 +327,14 @@ END
|
|||||||
|
|
||||||
复制 Linux 内核并分配 initramfs 给 EFI 系统分区:
|
复制 Linux 内核并分配 initramfs 给 EFI 系统分区:
|
||||||
|
|
||||||
```livescript
|
```
|
||||||
$ cp $(find /fc28/lib/modules -maxdepth 2 -name 'vmlinuz' | grep -m 1 $DEFAULT_VER) $HOME/esp/linux/vmlinuz-$DEFAULT_VER
|
$ cp $(find /fc28/lib/modules -maxdepth 2 -name 'vmlinuz' | grep -m 1 $DEFAULT_VER) $HOME/esp/linux/vmlinuz-$DEFAULT_VER
|
||||||
$ cp $(find /fc28/boot -name 'init*' | grep -m 1 $DEFAULT_VER) $HOME/esp/linux/initramfs-$DEFAULT_VER.img
|
$ cp $(find /fc28/boot -name 'init*' | grep -m 1 $DEFAULT_VER) $HOME/esp/linux/initramfs-$DEFAULT_VER.img
|
||||||
```
|
```
|
||||||
|
|
||||||
我们最终的目录布局应该看起来像下面的样子:
|
我们最终的目录布局应该看起来像下面的样子:
|
||||||
|
|
||||||
```livescript
|
```
|
||||||
esp
|
esp
|
||||||
├── efi
|
├── efi
|
||||||
│ └── boot
|
│ └── boot
|
||||||
@ -347,17 +345,17 @@ esp
|
|||||||
└── vmlinuz-4.18.18-200.fc28.x86_64
|
└── vmlinuz-4.18.18-200.fc28.x86_64
|
||||||
```
|
```
|
||||||
|
|
||||||
使用 QEMU 去使用我们的 EFI 系统分区,我们需要去创建一个小的 “uefi.img” 磁盘镜像来包含它,然后将它连接到 QEMU 作为主引导驱动器。
|
要让 QEMU 去使用我们的 EFI 系统分区,我们需要去创建一个小的 `uefi.img` 磁盘镜像来包含它,然后将它连接到 QEMU 作为主引导驱动器。
|
||||||
|
|
||||||
开始安装必需的工具:
|
开始安装必需的工具:
|
||||||
|
|
||||||
```livescript
|
```
|
||||||
$ dnf install -y parted dosfstools
|
$ dnf install -y parted dosfstools
|
||||||
```
|
```
|
||||||
|
|
||||||
现在创建 “uefi.img” 文件,并将 “esp” 目录中文件复制进去:
|
现在创建 `uefi.img` 文件,并将 `esp` 目录中的文件复制进去:
|
||||||
|
|
||||||
```livescript
|
```
|
||||||
$ ESP_SIZE=$(du -ks $HOME/esp | cut -f 1)
|
$ ESP_SIZE=$(du -ks $HOME/esp | cut -f 1)
|
||||||
$ dd if=/dev/zero of=$HOME/uefi.img count=$((${ESP_SIZE}+5000)) bs=1KiB
|
$ dd if=/dev/zero of=$HOME/uefi.img count=$((${ESP_SIZE}+5000)) bs=1KiB
|
||||||
$ UEFI_DEV=$(losetup --show -f $HOME/uefi.img)
|
$ UEFI_DEV=$(losetup --show -f $HOME/uefi.img)
|
||||||
@ -370,54 +368,55 @@ $ umount $HOME/mnt
|
|||||||
$ losetup -d ${UEFI_DEV}
|
$ losetup -d ${UEFI_DEV}
|
||||||
```
|
```
|
||||||
|
|
||||||
> 注意:在物理计算机上,你只需要从 “esp” 目录中复制文件到计算机上已存在的 EFI 系统分区中。你不需要使用 “uefi.img” 文件去引导物理计算机。
|
> 注意:在物理计算机上,你只需要从 `esp` 目录中复制文件到计算机上已存在的 EFI 系统分区中。你不需要使用 `uefi.img` 文件去引导物理计算机。
|
||||||
>
|
>
|
||||||
> 注意:在一个物理计算机上,如果文件名已存在,你可以重命名 “bootx64.efi” 文件,如果你重命名了它,就需要去编辑计算机的 BIOS 设置,并添加重命令后的 efi 文件到引导列表中。
|
> 注意:在一个物理计算机上,如果文件名已存在,你可以重命名 `bootx64.efi` 文件,如果你重命名了它,就需要去编辑计算机的 BIOS 设置,并添加重命令后的 efi 文件到引导列表中。
|
||||||
|
|
||||||
接下来我们需要去安装 qemu 包:
|
接下来我们需要去安装 qemu 包:
|
||||||
|
|
||||||
```livescript
|
```
|
||||||
$ dnf install -y qemu-system-x86
|
$ dnf install -y qemu-system-x86
|
||||||
```
|
```
|
||||||
|
|
||||||
允许 QEMU 访问我们在本教程“初始化配置”一节中创建的网桥:
|
允许 QEMU 访问我们在本教程“初始化配置”一节中创建的网桥:
|
||||||
|
|
||||||
```livescript
|
```
|
||||||
$ echo 'allow br0' > /etc/qemu/bridge.conf
|
$ echo 'allow br0' > /etc/qemu/bridge.conf
|
||||||
```
|
```
|
||||||
|
|
||||||
创建一个 “OVMF_VARS.fd” 镜像的副本去保存我们虚拟机的持久 BIOS 配置:
|
创建一个 `OVMF_VARS.fd` 镜像的副本去保存我们虚拟机的持久 BIOS 配置:
|
||||||
|
|
||||||
```livescript
|
```
|
||||||
$ cp /usr/share/edk2/ovmf/OVMF_VARS.fd $HOME
|
$ cp /usr/share/edk2/ovmf/OVMF_VARS.fd $HOME
|
||||||
```
|
```
|
||||||
|
|
||||||
现在,启动虚拟机:
|
现在,启动虚拟机:
|
||||||
|
|
||||||
```livescript
|
```
|
||||||
$ qemu-system-x86_64 -machine accel=kvm -nographic -m 1024 -drive if=pflash,format=raw,unit=0,file=/usr/share/edk2/ovmf/OVMF_CODE.fd,readonly=on -drive if=pflash,format=raw,unit=1,file=$HOME/OVMF_VARS.fd -drive if=ide,format=raw,file=$HOME/uefi.img -net bridge,br=br0 -net nic,model=virtio
|
$ qemu-system-x86_64 -machine accel=kvm -nographic -m 1024 -drive if=pflash,format=raw,unit=0,file=/usr/share/edk2/ovmf/OVMF_CODE.fd,readonly=on -drive if=pflash,format=raw,unit=1,file=$HOME/OVMF_VARS.fd -drive if=ide,format=raw,file=$HOME/uefi.img -net bridge,br=br0 -net nic,model=virtio
|
||||||
```
|
```
|
||||||
|
|
||||||
如果一切顺利,你将看到类似下图所示的结果:
|
如果一切顺利,你将看到类似下图所示的结果:
|
||||||
|
|
||||||
![][10]
|
![][10]
|
||||||
你可以使用 “shutdown” 命令关闭虚拟机回到我们的服务器上:
|
|
||||||
|
|
||||||
```livescript
|
你可以使用 `shutdown` 命令关闭虚拟机回到我们的服务器上:
|
||||||
|
|
||||||
|
```
|
||||||
$ sudo shutdown -h now
|
$ sudo shutdown -h now
|
||||||
```
|
```
|
||||||
|
|
||||||
> 注意:如果出现了错误或虚拟机挂住了,你可能需要启动一个新的 SSH 会话去连接服务器,使用 “kill” 命令去终止 “qemu-system-x86_64” 进程。
|
> 注意:如果出现了错误或虚拟机挂住了,你可能需要启动一个新的 SSH 会话去连接服务器,使用 `kill` 命令去终止 `qemu-system-x86_64` 进程。
|
||||||
|
|
||||||
### 镜像中添加包
|
### 镜像中添加包
|
||||||
|
|
||||||
镜像中添加包应该是一个很简单的问题,在服务器上 chroot 进镜像,然后运行 “dnf install <package_name>”。
|
镜像中添加包应该是一个很简单的问题,在服务器上 `chroot` 进镜像,然后运行 `dnf install <package_name>`。
|
||||||
|
|
||||||
在网络引导镜像中并不限制你能安装什么包。一个完整的图形化安装应该能够完美地工作。
|
在网络引导镜像中并不限制你能安装什么包。一个完整的图形化安装应该能够完美地工作。
|
||||||
|
|
||||||
下面是一个如何将最小化安装的网络引导镜像变成完整的图形化安装的示例:
|
下面是一个如何将最小化安装的网络引导镜像变成完整的图形化安装的示例:
|
||||||
|
|
||||||
```livescript
|
```
|
||||||
$ for i in dev dev/pts dev/shm proc sys run; do mount -o bind /$i /fc28/$i; done
|
$ for i in dev dev/pts dev/shm proc sys run; do mount -o bind /$i /fc28/$i; done
|
||||||
$ chroot /fc28 /usr/bin/bash --login
|
$ chroot /fc28 /usr/bin/bash --login
|
||||||
$ dnf -y groupinstall "Fedora Workstation"
|
$ dnf -y groupinstall "Fedora Workstation"
|
||||||
@ -430,9 +429,9 @@ $ logout
|
|||||||
$ for i in run sys proc dev/shm dev/pts dev; do umount /fc28/$i; done
|
$ for i in run sys proc dev/shm dev/pts dev; do umount /fc28/$i; done
|
||||||
```
|
```
|
||||||
|
|
||||||
可选,你可能希望去启用 “liveuser” 用户的自动登陆:
|
可选地,你可能希望去启用 `liveuser` 用户的自动登录:
|
||||||
|
|
||||||
```livescript
|
```
|
||||||
$ sed -i '/daemon/a AutomaticLoginEnable=true' /fc28/etc/gdm/custom.conf
|
$ sed -i '/daemon/a AutomaticLoginEnable=true' /fc28/etc/gdm/custom.conf
|
||||||
$ sed -i '/daemon/a AutomaticLogin=liveuser' /fc28/etc/gdm/custom.conf
|
$ sed -i '/daemon/a AutomaticLogin=liveuser' /fc28/etc/gdm/custom.conf
|
||||||
```
|
```
|
||||||
@ -444,7 +443,7 @@ via: https://fedoramagazine.org/how-to-build-a-netboot-server-part-1/
|
|||||||
作者:[Gregory Bartholomew][a]
|
作者:[Gregory Bartholomew][a]
|
||||||
选题:[lujun9972][b]
|
选题:[lujun9972][b]
|
||||||
译者:[qhwdw](https://github.com/qhwdw)
|
译者:[qhwdw](https://github.com/qhwdw)
|
||||||
校对:[校对者ID](https://github.com/校对者ID)
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
@ -1,28 +1,30 @@
|
|||||||
Bash 环境变量的那些事
|
Bash 环境变量的那些事
|
||||||
======
|
======
|
||||||
|
> 初学者可以在此教程中了解环境变量。
|
||||||
|
|
||||||
![](https://www.linux.com/sites/lcom/files/styles/rendered_file/public/wynand-van-poortvliet-40467-unsplash.jpg?itok=tr6Eb4N0)
|
![](https://www.linux.com/sites/lcom/files/styles/rendered_file/public/wynand-van-poortvliet-40467-unsplash.jpg?itok=tr6Eb4N0)
|
||||||
|
|
||||||
bash 变量,尤其是讨厌的环境变量,已经是一个老生常谈的话题了。我们也更应该对它有一个详细的了解,让它为我们所用。
|
bash 变量,尤其是讨厌的*环境变量*,已经是一个老生常谈的话题了。我们也更应该对它有一个详细的了解,让它为我们所用。
|
||||||
|
|
||||||
下面就打开终端,开始吧。
|
下面就打开终端,开始吧。
|
||||||
|
|
||||||
### 环境变量
|
### 环境变量
|
||||||
|
|
||||||
`HOME` 除了是你脱下帽子惬意休息的地方,同时也是 Linux 中的一个变量,它是当前用户主目录的路径:
|
`HOME` (LCTT 译注:双关语)除了是你脱下帽子惬意休息的地方,同时也是 Linux 中的一个变量,它是当前用户主目录的路径:
|
||||||
|
|
||||||
```
|
```
|
||||||
echo $HOME
|
echo $HOME
|
||||||
```
|
```
|
||||||
|
|
||||||
以上这个命令会显示当前用户的主目录路径,通常都在 `/home/` 下。
|
以上这个命令会显示当前用户的主目录路径,通常都在 `/home/<your username>` 下。
|
||||||
|
|
||||||
顾名思义,一个变量的值并不是固定的。实际上,Linux 系统中每一个用户的 `HOME` 变量都是不一样的,当然你也可以这样自行更改 `HOME` 变量的值:
|
顾名思义,变量的值是可以根据上下文变化的。实际上,Linux 系统中每一个用户的 `HOME` 变量都是不一样的,当然你也可以这样自行更改 `HOME` 变量的值:
|
||||||
|
|
||||||
```
|
```
|
||||||
HOME=/home/<your username>/Documents
|
HOME=/home/<your username>/Documents
|
||||||
```
|
```
|
||||||
|
|
||||||
以上这个命令将会把 `HOME` 变量设置为 `/home/<your username>/Documents` 目录。
|
以上这个命令将会把 `HOME` 变量设置为你的 `Documents` 目录。
|
||||||
|
|
||||||
其中有三点需要留意:
|
其中有三点需要留意:
|
||||||
|
|
||||||
@ -45,7 +47,7 @@ $ echo $PATH
|
|||||||
/usr/local/sbin:/usr/local/bin:/usr/bin:/usr/sbin:/bin:/sbin
|
/usr/local/sbin:/usr/local/bin:/usr/bin:/usr/sbin:/bin:/sbin
|
||||||
```
|
```
|
||||||
|
|
||||||
每两个目录之间使用冒号(`:`)分隔。如果某个应用程序的所在目录不在 `PATH` 变量中,那么运行的时候就需要声明应用程序的目录让 shell 能够找到。
|
每两个目录之间使用冒号 `:` 分隔。如果某个应用程序的所在目录不在 `PATH` 变量中,那么运行的时候就需要声明应用程序的目录让 shell 能够找到。
|
||||||
|
|
||||||
```
|
```
|
||||||
/home/<user name>/bin/my_program.sh
|
/home/<user name>/bin/my_program.sh
|
||||||
@ -67,9 +69,9 @@ PATH=$PATH:$HOME/bin
|
|||||||
|
|
||||||
然后 `/home/<user name>/bin/` 目录就会出现在 `PATH` 变量中了。但正如之前所说,这个变更只会在当前的 shell 生效,当前的 shell 一旦关闭,环境变量的值就又恢复原状了。
|
然后 `/home/<user name>/bin/` 目录就会出现在 `PATH` 变量中了。但正如之前所说,这个变更只会在当前的 shell 生效,当前的 shell 一旦关闭,环境变量的值就又恢复原状了。
|
||||||
|
|
||||||
如果要让变更对当前用户持续生效,就不能在 shell 中直接执行对应的变更,而是应该将这些变更操作卸载每次启动 shell 时都会运行的文件当中。这个文件就是当前用户主目录中的 `.bashrc` 文件。文件名前面的点号表明这是一个隐藏文件,执行普通的 `ls` 命令是不会将这个文件显示出来的,但只要在 `ls` 命令中加入 `-a` 参数就可以看到这个文件了。
|
如果要让变更对当前用户持续生效,就不能在 shell 中直接执行对应的变更,而是应该将这些变更操作写在每次启动 shell 时都会运行的文件当中。这个文件就是当前用户主目录中的 `.bashrc` 文件。文件名前面的点号表明这是一个隐藏文件,执行普通的 `ls` 命令是不会将这个文件显示出来的,但只要在 `ls` 命令中加入 `-a` 参数就可以看到这个文件了。
|
||||||
|
|
||||||
你可以使用诸如 [kate][1]、[gedit][2]、[nano][3] 或者 [vim][4] 这些文本编辑器来打开 `.bashrc` 文件(但不要用 LibreOffice Writer,它是一个文字处理软件,跟前面几个文字编辑器并不一个量级的东西)。打开 `.bashrc` 文件之后,你会看见里面放置了一些 shell 命令,是用于为当前用户设置环境的。
|
你可以使用诸如 [kate][1]、[gedit][2]、[nano][3] 或者 [vim][4] 这些文本编辑器来打开 `.bashrc` 文件(但不要用 LibreOffice Writer,它是一个文字处理软件,跟前面几个文字编辑器完全不同)。打开 `.bashrc` 文件之后,你会看见里面放置了一些 shell 命令,是用于为当前用户设置环境的。
|
||||||
|
|
||||||
在文件的末尾添加新行并输入以下内容:
|
在文件的末尾添加新行并输入以下内容:
|
||||||
|
|
||||||
@ -97,13 +99,13 @@ source .bashrc
|
|||||||
new_variable="Hello"
|
new_variable="Hello"
|
||||||
```
|
```
|
||||||
|
|
||||||
然后可以用一下的方式读取到已定义变量的值:
|
然后可以用以下的方式读取到已定义变量的值:
|
||||||
|
|
||||||
```
|
```
|
||||||
echo $new_variable
|
echo $new_variable
|
||||||
```
|
```
|
||||||
|
|
||||||
程序的正常工作离不开各种变量,例如要将某个选项设置为 on,又或者让程序找到所需的代码库,都需要使用变量。在 bash 中运行程序的时候会生成一个子 shell,这个子 shell 和执行原程序的父 shell 并不是完全一样的,只是继承了父 shell 的部分内容,而且默认是不继承父 shell 中的变量的。因为变量默认情况下是局部变量,出于安全原因,一个 shell 中的局部变量不会被另一个 shell 读取到,即使是子 shell 也不可以。
|
程序的正常工作离不开各种变量,例如要将某个选项设置为打开,又或者让程序找到所需的代码库,都需要使用变量。在 bash 中运行程序的时候会生成一个子 shell,这个子 shell 和执行原程序的父 shell 并不是完全一样的,只是继承了父 shell 的部分内容,而且默认是不继承父 shell 中的变量的。因为变量默认情况下是局部变量,出于安全原因,一个 shell 中的局部变量不会被另一个 shell 读取到,即使是子 shell 也不可以。
|
||||||
|
|
||||||
下面举一个例子。首先定义一个变量:
|
下面举一个例子。首先定义一个变量:
|
||||||
|
|
||||||
@ -198,7 +200,7 @@ via: https://www.linux.com/blog/learn/2018/12/bash-variables-environmental-and-o
|
|||||||
作者:[Paul Brown][a]
|
作者:[Paul Brown][a]
|
||||||
选题:[lujun9972][b]
|
选题:[lujun9972][b]
|
||||||
译者:[HankChow](https://github.com/HankChow)
|
译者:[HankChow](https://github.com/HankChow)
|
||||||
校对:[校对者ID](https://github.com/校对者ID)
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue
Block a user