From eef47403bf13f34fba003404cbf9694462c9fb4f Mon Sep 17 00:00:00 2001 From: Brooke Lau Date: Tue, 7 Jan 2020 00:46:38 +0800 Subject: [PATCH 01/20] tmp --- ... Lessons learned from programming in Go.md | 191 +++++++++--------- 1 file changed, 96 insertions(+), 95 deletions(-) diff --git a/sources/tech/20191210 Lessons learned from programming in Go.md b/sources/tech/20191210 Lessons learned from programming in Go.md index 0c9aeea2e2..93e9af3398 100644 --- a/sources/tech/20191210 Lessons learned from programming in Go.md +++ b/sources/tech/20191210 Lessons learned from programming in Go.md @@ -1,71 +1,70 @@ -[#]: collector: (lujun9972) -[#]: translator: (lxbwolf) -[#]: reviewer: ( ) -[#]: publisher: ( ) -[#]: url: ( ) -[#]: subject: (Lessons learned from programming in Go) -[#]: via: (https://opensource.com/article/19/12/go-common-pitfalls) -[#]: author: (Eduardo Ferreira https://opensource.com/users/edufgf) +[#]: collector: "lujun9972" +[#]: translator: "lxbwolf" +[#]: reviewer: " " +[#]: publisher: " " +[#]: url: " " +[#]: subject: "Lessons learned from programming in Go" +[#]: via: "https://opensource.com/article/19/12/go-common-pitfalls" +[#]: author: "Eduardo Ferreira https://opensource.com/users/edufgf" -Lessons learned from programming in Go +用 Go 编程受到的启发 ====== -Prevent future concurrent processing headaches by learning how to -address these common pitfalls. +通过学习如何定位并发处理的陷阱来避免未来处理这些问题时的困境。 ![Goland gopher illustration][1] -When you are working with complex distributed systems, you will likely come across the need for concurrent processing. At [Mode.net][2], we deal daily with real-time, fast and resilient software. Building a global private network that dynamically routes packets at the millisecond scale wouldn’t be possible without a highly concurrent system. This dynamic routing is based on the state of the network and, while there are many parameters to consider here, our focus is on link [metrics][3]. In our context, link metrics can be anything related to the status or current properties of a network link (e.g.: link latency). +在复杂的分布式系统进行任务处理时,你通常会需要进行并发的操作。[Mode.net][2] 公司系统每天要处理实时、快速和灵活的以毫秒为单位动态路由数据包的全球专用网络和数据,需要高度并发的系统。他们的动态路由是基于网络状态的,而这个过程需要考虑众多因素,我们只考虑关系链的监控。在我们的环境中,调用关系链监控可以是任何跟网络调用关系链有关的状态和当前属性(如连接延迟)。 -### Concurrent probing for link metrics +### 并发探测链接指标 -[H.A.L.O.][4] (Hop-by-Hop Adaptive Link-State Optimal Routing), our dynamic routing algorithm relies partially on link metrics to compute its routing table. Those metrics are collected by an independent component that sits on each [PoP][5] (Point of Presence). PoPs are machines that represent a single routing entity in our networks, connected by links and spread around multiple locations shaping our network. This component probes neighboring machines using network packets, and those neighbors will bounce back the initial probe. Link latency values can be derived from the received probes. Because each PoP has more than one neighbor, the nature of such a task is intrinsically concurrent: we need to measure latency for each neighboring link in real-time. We can’t afford sequential processing; each probe must be processed as soon as possible in order to compute this metric. +[H.A.L.O.][4] (Hop-by-Hop Adaptive Link-State Optimal Routing,译注:逐跳自适应链路状态最佳路由), 我们的动态路由算法部分依赖于链路度量来计算路由表。 这些指标由位于每个PoP(译注:存活节点)上的独立组件收集。PoP是表示我们的网络中单个路由实体的机器,通过链路连接并分布在我们的网络拓扑中的各个位置。某个组件使用网络数据包探测周围的机器,周围的机器回复数据包给前者。从接收到的探测包中可以获得链路延迟。由于每个 PoP 都有多个临近节点,所以这种探测任务实质上是并发的:我们需要实时测量每个临近连接点的延迟。我们不能串行地处理;为了计算这个指标,必须尽快处理每个探针。 ![latency computation graph][6] -### Sequence numbers and resets: A reordering situation +### 序列号和重置:一个记录场景 -Our probing component exchanges packets and relies on sequence numbers for packet processing. This aims to avoid processing of packet duplication or out-of-order packets. Our first implementation relied on a special sequence number 0 to reset sequence numbers. Such a number was only used during initialization of a component. The main problem was that we were considering an increasing sequence number value that always started at 0. After the component restarts, packet reordering could happen, and a packet could easily replace the sequence number with the value that was being used before the reset. This meant that the following packets would be ignored until it reaches the sequence number that was in use just before the reset. +我们的探测组件互相发送和接收数据包并依靠序列号进行数据包处理。旨在避免处理重复的包或顺序被打乱的包。我们的第一个实现依靠特殊的序列号0来重置序列号。这个数字仅在组件初始化时使用。主要的问题是我们只考虑了始终从 0 开始递增的序列号。组件重启后,包的顺序可能会重新排列,某个包的序列号可能会轻易地被替换成重置之前使用过的值。这意味着,直到排到重置之前用到的值之前,它后面的包都会被忽略掉。 -### UDP handshake and finite state machine +### UDP 握手和有限状态机 -The problem here was proper agreement of a sequence number after a component restarts. There are a few ways to handle this and, after discussing our options, we chose to implement a 3-way handshake protocol with a clear definition of states. This handshake establishes sessions over links during initialization. This guarantees that nodes are communicating over the same session and using the appropriate sequence number for it. +这里的问题是重启前后的序列号是否一致。有几种方法可以解决这个问题,经过讨论,我们选择了实现一个带有清晰状态定义的三向交握协议。这个握手过程在初始化时通过链接建立 session。这样可以确保节点通过同一个 session 进行通信且使用了适当的序列号。 -To properly implement this, we have to define a finite state machine with clear states and transitions. This allows us to properly manage all corner cases for the handshake formation. +为了正确实现这个过程,我们必须顶一个一个有清晰状态和过渡的有限状态机。这样我们就可以正确管理握手过程中的所有极端情况。 ![finite state machine diagram][7] -Session IDs are generated by the handshake initiator. A full exchange sequence is as follows: +session ID 由握手的初始化程序生成。一个完整的交换顺序如下: - 1. The sender sends out a **SYN (ID)*** *packet. - 2. The receiver stores the received **ID** and sends a **SYN-ACK (ID)**. - 3. The sender receives the **SYN-ACK (ID) *_and sends out an **ACK (ID)**._ *It also starts sending packets starting with sequence number 0. - 4. The receiver checks the last received **ID*** _and accepts the **ACK (ID)**_ *if the ID matches. It also starts accepting packets with sequence number 0. + 1. sender 发送一个 **SYN (ID)** 数据包。 + 2. receiver 存储接收到的 **ID** 并发送一个 **SYN-ACK (ID)**. + 3. sender 接收到 **SYN-ACK (ID)** _并发送一个 **ACK (ID)**_。它还发送一个从序列号 0 开始的数据包。 + 4. receiver 检查最后接收到的 **ID**,如果 ID 匹配,_则接受 **ACK (ID)**_。它还开始接受序列号为 0 的数据包。 -### Handling state timeouts +### 处理状态超时 -Basically, at each state, you need to handle, at most, three types of events: link events, packet events, and timeout events. And those events show up concurrently, so here you have to handle concurrency properly. +Basically, at each state, you need to handle, at most, three types of events: link events, packet events, and timeout events. And those events show up concurrently, so here you have to handle concurrency properly.基本上,每种状态下你都需要处理最多三种类型的事件:链接事件、数据包事件和超时事件。这些事件会并发地出现,因此你必须正确处理并发。 - * Link events are either link up or link down updates. This can either initiate a link session or break an existing session. - * Packet events are control packets **(SYN/SYN-ACK/ACK)** or just probe responses. - * Timeout events are the ones triggered after a scheduled timeout expires for the current session state. + * 链接事件包括连接和断开,连接时会初始化一个链接 session,断开时会断开一个已建立的 seesion。 + * 数据包事件是控制数据包 **(SYN/SYN-ACK/ACK)** 或只是探测响应。 + * 超时事件在当前 session 状态的预定超时时间到期后触发。 -The main challenge here is how to handle concurrent timeout expiration and other events. And this is where one can easily fall into the traps of deadlocks and race conditions. +这里面临的最主要的问题是如何处理并发超时到期和其他事件。这里很容易陷入死锁和资源竞争的陷阱。 -### A first approach +### 第一种方法 -The language used for this project is [Golang][8]. It does provide native synchronization mechanisms such as native channels and locks and is able to spin lightweight threads for concurrent processing. +本项目使用的语言是 [Golang][8]. 它确实提供了原生的同步机制,如自带的 channel 和锁,并且能够使用轻量级线程来进行并发处理。 ![gophers hacking together][9] gophers hacking together -You can start first by designing a structure that represents our **Session** and **Timeout Handlers**. +首先,你可以设计两个分别表示我们的 **Session** 和 **Timeout Handlers** 的结构体。 -``` +```go type Session struct {     State SessionState     Id SessionId   @@ -80,21 +79,21 @@ type TimeoutHandler struct {   } ``` -**Session** identifies the connection session, with the session ID, neighboring link IP, and the current session state. +**Session** 标识连接 session,内有表示 session ID、临近的连接点的 IP 和当前 session 状态的字段。 -**TimeoutHandler** holds the callback function, the session for which it should run, the duration, and a pointer to the scheduled timer. +**TimeoutHandler** 包含回调函数、对应的 session、持续时间和指向调度计时器的 timer 指针。 -There is a global map that will store, per neighboring link session, the scheduled timeout handler. +每一个临近连接点的 session 都包含一个保存调度 `TimeoutHandler` 的全局 map。 ``` `SessionTimeout map[Session]*TimeoutHandler` ``` -Registering and canceling a timeout is achieved by the following methods: +下面方法注册和取消超时: -``` +```go // schedules the timeout callback function.   func (timeout* TimeoutHandler) Register() {     timeout.timer = time.AfterFunc(time.Duration(timeout.duration) * time.Second, func() {   @@ -110,10 +109,10 @@ func (timeout* TimeoutHandler) Cancel() {   } ``` -For the timeouts creation and storage, you can use a method like the following: +你可以使用类似下面的方法来创建和存储超时: -``` +```go func CreateTimeoutHandler(callback func(Session), session Session, duration int) *TimeoutHandler {     if sessionTimeout[session] == nil {       sessionTimeout[session] := new(TimeoutHandler)   @@ -127,12 +126,12 @@ func CreateTimeoutHandler(callback func(Session), session Session, duration int) } ``` -Once the timeout handler is created and registered, it runs the callback after _duration_ seconds have elapsed. However, some events will require you to reschedule a timeout handler (as it happens at **SYN** state — every 3 seconds). +超时 handler 创建后,会在经过了设置的 _duration_ 时间(秒)后执行回调函数。然而,有些事件会使你重新调度一个超时 handler(与 **SYN** 状态时的处理一样 — 每 3 秒一次)。 -For that, you can have the callback rescheduling a new timeout: +为此,你可以让回调函数重新调度一次超时: -``` +```go func synCallback(session Session) {     sendSynPacket(session) @@ -144,34 +143,34 @@ func synCallback(session Session) {   } ``` -This callback reschedules itself in a new timeout handler and updates the global **sessionTimeout** map. +这次回调在新的超时 handler 中重新调度自己,并更新全局 map **sessionTimeout**。 -### **Data race and references** +### 数据竞争和引用 -Your solution is ready. One simple test is to check that a timeout callback is executed after the timer has expired. To do this, register a timeout, sleep for its duration, and then check whether the callback actions were done. After the test is executed, it is a good idea to cancel the scheduled timeout (as it reschedules), so it won’t have side effects between tests. +你的解决方案已经有了。可以通过检查计时器到期后超时回调是否执行来进行一个简单的测试。为此,注册一个超时,在 *duration* 时间内 sleep,然后检查是否执行的回调的处理。执行这个测试后,最好取消预定的超时时间(因为它会重新调度),这样才不会在下次测试时产生副作用。 -Surprisingly, this simple test found a bug in the solution. Canceling timeouts using the cancel method was just not doing its job. The following order of events would cause a data race condition: +令人惊讶的是,这个简单的测试发现了这个解决方案中的一个 bug。使用 cancel 方法来取消超时并没有正确处理。以下顺序的事件会导致数据资源竞争: - 1. You have one scheduled timeout handler. - 2. Thread 1: -a) You receive a control packet, and you now want to cancel the registered timeout and move on to the next session state. (e.g. received a **SYN-ACK** **after you sent a **SYN**). -b) You call **timeout.Cancel()**, which calls a **timer.Stop()**. (Note that a Golang timer stop doesn’t prevent an already expired timer from running.) - 3. Thread 2: -a) Right before that cancel call, the timer has expired, and the callback was about to execute. -b) The callback is executed, it schedules a new timeout and updates the global map. - 4. Thread 1: -a) Transitions to a new session state and registers a new timeout, updating the global map. + 1. 你有一个已调度的超时 handler。 + 2. 线程 1: +a) 你接收到一个控制数据包,现在你要取消已注册的超时并切换到下一个 session 状态(如 发送 **SYN** 后接收到一个 **SYN-ACK**) +b) 你调用了 **timeout.Cancel()**,这个函数调用了 **timer.Stop()**。(请注意,Golang 计时器的 stop 不会终止一个已过期的计时器。) + 3. 线程 2: +a) 在调用 cancel 之前,计时器已过期,回调即将执行。 +b) 执行回调,它调度一次新的超时并更新全局 map。 + 4. 线程 1: +a) 切换到新的 session 状态并注册新的超时,更新全局 map。 -Both threads were updating the timeout map concurrently. The end result is that you failed to cancel the registered timeout, and then you also lost the reference to the rescheduled timeout done by thread 2. This results in a handler that keeps executing and rescheduling for a while, doing unwanted behavior. +两个线程同时更新超时 map。最终结果是你无法取消注册的超时,然后你也会丢失对线程 2 重新调度的超时的引用。这导致 handler 在一段时间内持续执行和重新调度,出现非预期行为。 -### When locking is not enough +### 锁也解决不了问题 -Using locks also doesn’t fix the issue completely. If you add locks before processing any event and before executing a callback, it still doesn’t prevent an expired callback to run: +使用锁也不能完全解决问题。如果你在处理所有事件和执行回调之前加锁,它仍然不能阻止一个过期的回调运行: -``` +```go func (timeout* TimeoutHandler) Register() {     timeout.timer = time.AfterFunc(time.Duration(timeout.duration) * time._Second_, func() {       stateLock.Lock()   @@ -182,26 +181,26 @@ func (timeout* TimeoutHandler) Register() {   } ``` -The difference now is that the updates in the global map are synchronized, but this doesn’t prevent the callback from running after you call the **timeout.Cancel() **— This is the case if the scheduled timer expired but didn’t grab the lock yet. You should again lose reference to one of the registered timeouts. +现在的区别就是全局 map 的更新是同步的,但是这还是不能阻止在你调用 **timeout.Cancel() ** 后回调的执行 — 这种情况出现在调度计时器过期了但是还没有拿到锁的时候。你还是会丢失一个已注册的超时的引用。 -### Using cancellation channels +### 使用取消 channel -Instead of relying on golang’s **timer.Stop()**, which doesn’t prevent an expired timer to execute, you can use cancellation channels. +你可以使用取消 channel,而不必依赖不能阻止到期的计时器执行的 golang 函数 **timer.Stop()**。 -It is a slightly different approach. Now you won’t do a recursive re-scheduling through callbacks; instead, you register an infinite loop that waits for cancellation signals or timeout events. +这是一个略有不同的方法。现在你可以不用再通过回调进行递归地重新调度;而是注册一个死循环,这个循环接收到取消信号或超时事件是终止。 -The new **Register()** spawns a new go thread that runs your callback after a timeout and schedules a new timeout after the previous one has been executed. A cancellation channel is returned to the caller to control when the loop should stop. +新的 **Register()** 产生一个新的 go 协程,这个协程在在超时后执行你的回调,并在前一个超时执行后调度新的超时。返回给调用发一个取消 channel,用来控制循环的终止。 -``` +```go func (timeout *TimeoutHandler) Register() chan struct{} {     cancelChan := make(chan struct{})         go func () {       select {   -    case _ = <\- cancelChan:   +    case _ = <- cancelChan:         return   -    case _ = <\- time.AfterFunc(time.Duration(timeout.duration) * time.Second):   +    case _ = <- time.AfterFunc(time.Duration(timeout.duration) * time.Second):         func () {           stateLock.Lock()           defer stateLock.Unlock() @@ -218,23 +217,23 @@ func (timeout* TimeoutHandler) Cancel() {     if timeout.cancelChan == nil {       return     }   -  timeout.cancelChan <\- struct{}{}   +  timeout.cancelChan <- struct{}{}   } ``` -This approach gives you a cancellation channel for each timeout you register. A cancel call sends an empty struct to the channel and triggers the cancellation. However, this doesn’t resolve the previous issue; the timeout can expire right before you call cancel over the channel, and before the lock is grabbed by the timeout thread. +这个方法提供了你注册的所有超时的取消 channel。对 cancel 的一次调用向 channel 发送一个空结构体并触发取消操作。然而,这并不能解决前面的问题;可能在你通过 channel 调用 cancel 超时线程还没有拿到锁之前,超时时间就已经到了。 -The solution here is to check the cancellation channel inside the timeout scope after you grab the lock. +这里的解决方案是,在拿到锁之后,检查一下超时范围内的取消 channel。 -``` -  case _ = <\- time.AfterFunc(time.Duration(timeout.duration) * time.Second):   +```go +  case _ = <- time.AfterFunc(time.Duration(timeout.duration) * time.Second):       func () {         stateLock.Lock()         defer stateLock.Unlock()               select {   -      case _ = <\- handler.cancelChan:   +      case _ = <- handler.cancelChan:           return         default:           timeout.callback(timeout.session)   @@ -243,47 +242,49 @@ The solution here is to check the cancellation channel inside the timeout scope   } ``` -Finally, this guarantees that the callback is only executed after you grab the lock and no cancellation was triggered. +最终,这可以确保在拿到锁之后执行回调,不会触发取消操作。 -### Beware of deadlocks +### 小心死锁 -This solution seems to work; however, there is one hidden pitfall here: [deadlocks][10]. +这个解决方案看起来有效;但是还是有个隐患:[死锁][10]。 -Please read the code above again and try to find it yourself. Think of concurrent calls to any of the methods described. +请阅读上面的代码,试着自己找到它。考虑下描述的所有函数的并发调用。 -The last problem here is with the cancellation channel itself. We made it an unbuffered channel, which means that sending is a blocking call. Once you call cancel in a timeout handler, you only proceed once that handler is canceled. The problem here is when you have multiple calls to the same cancelation channel, where a cancel request is only consumed once. And this can easily happen if concurrent events were to cancel the same timeout handler, like a link down or control packet event. This results in a deadlock situation, possibly bringing the application to a halt. +这里的问题在取消 channel 本身。我们创建的是无缓冲 channel,即发送是阻塞调用。当你在一个超时 handler 中调用取消函数时,只有在该 handler 被取消后才能继续处理。问题出现在,当你有多个调用请求到同一个取消 channel 时,这时一个取消请求只被处理一次。当多个事件同时取消同一个超时 handler 时,如链接断开或控制包事件,很容易出现这种情况。这会导致死锁,可能会使应用程序 halt。 ![gophers on a wire, talking][11] -Is anyone listening? +还有人吗? -By Trevor Forrey. Used with permission. +已获得 Trevor Forrey 授权。 -The solution here is to at least make the channel buffered by one, so sends are not always blocking, and also explicitly make the send non-blocking in case of concurrent calls. This guarantees the cancellation is sent once and won’t block the subsequent cancel calls. +这里的解决方案是创建 channel 时指定大小至少为 1,这样向 channel 发送数据就不会阻塞,也显式地使发送变成非阻塞的,避免了并发调用。这样可以确保取消操作只发送一次,并且不会则色后续的取消调用。 -``` +```go func (timeout* TimeoutHandler) Cancel() {     if timeout.cancelChan == nil {       return     }         select {   -  case timeout.cancelChan <\- struct{}{}:   +  case timeout.cancelChan <- struct{}{}:     default:       // can’t send on the channel, someone has already requested the cancellation.     }   } ``` -### Conclusion +### 总结 -You learned in practice how common mistakes can show up while working with concurrent code. Due to their non-deterministic nature, those issues can go easily undetected, even with extensive testing. Here are the three main problems we encountered in the initial implementation. +在实践中你学到了并发操作时出现的常见错误。由于其不确定性,即使进行大量的测试,也不容易发现这些问题。下面是我们在最初的实现中遇到的三个主要问题: -#### Updating shared data without synchronization +#### 在非同步的情况下更新共享数据 This seems like an obvious one, but it’s actually hard to spot if your concurrent updates happen in different locations. The result is data race, where multiple updates to the same data can cause update loss, due to one update overriding another. In our case, we were updating the scheduled timeout reference on the same shared map. (Interestingly, if Go detects a concurrent read/write on the same Map object, it throws a fatal error —you can try to run Go’s [data race detector][12]). This eventually results in losing a timeout reference and making it impossible to cancel that given timeout. Always remember to use locks when they are needed. + + ![gopher assembly line][13] don’t forget to synchronize gophers’ work @@ -317,18 +318,18 @@ via: https://opensource.com/article/19/12/go-common-pitfalls [a]: https://opensource.com/users/edufgf [b]: https://github.com/lujun9972 -[1]: https://opensource.com/sites/default/files/styles/image-full-size/public/lead-images/go-golang.png?itok=OAW9BXny (Goland gopher illustration) +[1]: https://opensource.com/sites/default/files/styles/image-full-size/public/lead-images/go-golang.png?itok=OAW9BXny "Goland gopher illustration" [2]: http://mode.net [3]: https://en.wikipedia.org/wiki/Metrics_%28networking%29 [4]: https://people.ece.cornell.edu/atang/pub/15/HALO_ToN.pdf [5]: https://en.wikipedia.org/wiki/Point_of_presence -[6]: https://opensource.com/sites/default/files/uploads/image2_0_3.png (latency computation graph) -[7]: https://opensource.com/sites/default/files/uploads/image3_0.png (finite state machine diagram) +[6]: https://opensource.com/sites/default/files/uploads/image2_0_3.png "latency computation graph" +[7]: https://opensource.com/sites/default/files/uploads/image3_0.png "finite state machine diagram" [8]: https://golang.org/ -[9]: https://opensource.com/sites/default/files/uploads/image4.png (gophers hacking together) +[9]: https://opensource.com/sites/default/files/uploads/image4.png "gophers hacking together" [10]: https://en.wikipedia.org/wiki/Deadlock -[11]: https://opensource.com/sites/default/files/uploads/image5_0_0.jpg (gophers on a wire, talking) +[11]: https://opensource.com/sites/default/files/uploads/image5_0_0.jpg "gophers on a wire, talking" [12]: https://golang.org/doc/articles/race_detector.html -[13]: https://opensource.com/sites/default/files/uploads/image6.jpeg (gopher assembly line) +[13]: https://opensource.com/sites/default/files/uploads/image6.jpeg "gopher assembly line" [14]: https://en.wikipedia.org/wiki/Monitor_%28synchronization%29#Condition_variables -[15]: https://opensource.com/sites/default/files/uploads/image7.png (gopher boot camp) +[15]: https://opensource.com/sites/default/files/uploads/image7.png "gopher boot camp" From 923e0da51ed798d772ab2a283d831cf23664657c Mon Sep 17 00:00:00 2001 From: Xingyu Wang Date: Tue, 7 Jan 2020 23:12:11 +0800 Subject: [PATCH 02/20] PRF MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit @BrunoJu 恭喜你完成了第一篇翻译贡献! --- ...es to accelerate your automation skills.md | 56 ++++++++++--------- 1 file changed, 29 insertions(+), 27 deletions(-) diff --git a/translated/tech/20191231 10 Ansible resources to accelerate your automation skills.md b/translated/tech/20191231 10 Ansible resources to accelerate your automation skills.md index f50180a6d7..f7805c35f0 100644 --- a/translated/tech/20191231 10 Ansible resources to accelerate your automation skills.md +++ b/translated/tech/20191231 10 Ansible resources to accelerate your automation skills.md @@ -1,43 +1,45 @@ [#]: collector: (lujun9972) [#]: translator: (BrunoJu) -[#]: reviewer: ( ) +[#]: reviewer: (wxy) [#]: publisher: ( ) [#]: url: ( ) [#]: subject: (10 Ansible resources to accelerate your automation skills) [#]: via: (https://opensource.com/article/19/12/ansible-resources) [#]: author: (James Farrell https://opensource.com/users/jamesf) -提升自动化技巧的10大Ansible资源 +提升自动化技巧的 10 篇 Ansible 文章 ====== -今年,准备好,用出色的Ansible自动化技能装备自己的技能包吧。 -![Gears above purple clouds][1] +> 今年,准备好,用出色的 Ansible 自动化技能装备自己的技能包吧。 -今年我关注了大量关于Ansible的文章,以下这些内容都值得每个人学习,无论是否是Ansible的新手。 +![](https://img.linux.net.cn/data/attachment/album/202001/07/231057fbtfjwrn29ficxj0.jpg) -这些文章值得大家标记为书签,或者设置个计划任务(亦或者是设置一个Tower/AWX job),用来提醒自己常读常新。 -如果你是Ansible的新手,那么就从这些文章开始着手吧: - * [_A quickstart guide to Ansible_][2] 拥有一些对新手非常有用的信息,同时还有一些更高级的话题。 - * [_10 Ansible modules you need to know_][3] 和 [_5 ops tasks to do with Ansible_][4] 这篇文章有每一位Ansible的管理者都应该熟悉并认真研习的一些最基础的Ansible功能 - * [_How to use Ansible to document procedures_][5] 这篇文章是对一些额外话题的纵览,我猜你一定会感到很有趣。 +今年我关注了大量关于 Ansible 的文章,以下这些内容都值得每个人学习,无论是否是 Ansible 的新手。 +这些文章值得大家标记为书签,或者设置个计划任务(亦或者是设置一个 Tower/AWX 任务),用来提醒自己常读常新。 -剩余的这些文章包含了更多高级的话题,比如Windows管理,测试、硬件、云和容器,甚至包括了一个案例研究,如何管理那些对技术有兴趣的孩子的需求。 +如果你是 Ansible 的新手,那么就从这些文章开始着手吧: -我希望你能像我一样好好享受Ansible带来的乐趣。不要停止学习哦! - 1. _[How Ansible brought peace to my home][6]_ 这个异想天开的案例,你能看到如何利用Ansible为孩子们快速部署一个新的笔记本(或者重装旧笔记本) - 2. _[Ansible for the Windows admin][7]_ 你知道Ansible也可以管理Windows的节点吗?这篇文章以部署一个IIS为案例,阐述了基础的Ansible服务器和Windows客户端的安装。 - 3. _[10 Ansible modules you need to know][3]_ 这是个学习你最应该知道的那些最常见最基础Ansible模块的好文章。执行命令、安装系统包和操作文件是许多有用的自动化工作的基础。 - 4. _[How to use Ansible to document procedures][5]_  Ansible的YAML文件易于阅读,因此它们可以被用于记录完成任务所需的手动步骤。这一特性可以帮助你调试与扩展,这令工作变得异常轻松。同时,这篇文章还包含关于测试和分析等Ansible相关主题的指导。 - 5. _[Using Testinfra with Ansible to verify server state][8]_ 测试环节是任何一个 CI/CD 开发型运维场景不可或缺的一部分。所以为什么不把测试Ansible的运行结果也纳入其中呢?这个有关于Ansible测试架构的入门级文章可以被用来帮助你配置。 - 6. _[Hardware bootstrapping with Ansible][9]_ 这个世界并不是完全已经被容器和虚拟机所占据。许多系统管理员仍然需要管理众多硬件资源。通过Ansible与PXE、DHCP以及其他一些技术的结合,你可以创建一个简单的管理框架,对硬件实施控制。 - 7. _[What you need to know about Ansible modules][10]_ 模块给Ansible带来了巨大的潜力,许多模块我们已经可以拿来利用。但如果没有你所需的模块,那你可以尝试给自己造一个。看看这篇文章吧,它能让你了解如何从零开始打造自己所需的模块。 - 8. _[5 ops tasks to do with Ansible][4]_ 这是另一个有关于如何使用Ansible来管理常见的系统操作任务的文章。这里描述了一系列可以取代命令行操作的Tower(AWX)的案例。 - 9. _[A quickstart guide to Ansible][2]_ 这篇文章是个可以下载的PDF文档。它可以作为一本随时拿来翻阅的手册。这篇文章的开头有助于初学者入门。同时,还包括了一些其他的研究领域,比如模块测试、系统管理任务和针对K8S对象的管理。 - 10. _[An Ansible reference guide, CI/CD with Ansible Tower and GitHub, and more news][11]_ 这是一篇每月进行总结更新的文章,充满了有趣的链接。话题包括了Ansible的基础内容、管理Netapp的E系列存储产品、调试、打补丁包和其他一些相关内容。文章中还包括了一些视频以及指导性文章的链接。击这里查看详情! +* 《[Ansible 快速入门指南][2]》拥有一些对新手非常有用的信息,同时还介绍了一些更高级的话题。 +* 《[你需要知道的 10 个 Ansible 模块][3]》和《[5 个 Ansible 运维任务][4]》(译文)这两篇文章有每一位 Ansible 管理员都应该熟悉并认真研习的一些最基础的 Ansible 功能。 +* 《[如何使用 Ansible 记录流程][5]》这篇文章是对一些额外话题的纵览,我猜你一定会感到很有趣。 +剩余的这些文章包含了更多高级的话题,比如 Windows 管理、测试、硬件、云和容器,甚至包括了一个案例研究,如何管理那些对技术有兴趣的孩子的需求。 -如果你也有一些你喜爱的Ansible文章,那请留言告诉我们吧。 +我希望你能像我一样好好享受 Ansible 带来的乐趣。不要停止学习哦! + +1. 《[Ansible 如何为我的家庭带来和平][6]》这个异想天开的案例,你能看到如何利用 Ansible 为孩子们快速部署一个新的笔记本(或者重装旧笔记本) +2. Taz Brown 和 Abner Malivert 的《[适用于 Windows 管理员的 Ansible][7]》:你知道 Ansible 也可以管理 Windows 的节点吗?这篇文章以部署一个 IIS 为案例,阐述了基础的 Ansible 服务器和 Windows 客户端的安装。 +3. Shashank Hegde 的《[你需要知道的 10 个 Ansible 模块][3]》是个学习你最应该知道的那些最常见、最基础的 Ansible 模块的好文章。运行命令、安装软件包和操作文件是许多有用的自动化工作的基础。 +4. Marco Bravo 的《[如何使用 Ansible 记录流程][5]》:Ansible 的 YAML 文件易于阅读,因此它们可以被用于记录完成任务所需的手动步骤。这一特性可以帮助你调试与扩展,这令工作变得异常轻松。同时,这篇文章还包含关于测试和分析等 Ansible 相关主题的指导。 +5. Clement Verna 的《[使用 Testinfra 和 Ansible 验证服务器状态][8]》(译文):测试环节是任何一个 CI/CD DevOps 流程不可或缺的一部分。所以为什么不把测试 Ansible 的运行结果也纳入其中呢?这个测试架构 Testinfra 的入门级文章可以帮助你检查配置结果。 +6. Mark Phillips 的《[Ansible 硬件起步][9]》:这个世界并不是完全已经被容器和虚拟机所占据。许多系统管理员仍然需要管理众多硬件资源。通过 Ansible 与一点 PXE、DHCP 以及其他技巧的结合,你可以创建一个方便的管理框架使硬件易于启动和运行。 +7. Jairo da Silva Junior 的《[你需要了解的关于 Ansible 模块的知识][10]》:模块给 Ansible 带来了巨大的潜力,已经有许多模块可以拿来利用。但如果没有你所需的模块,那你可以尝试给自己打造一个。看看这篇文章吧,它能让你了解如何从零开始打造自己所需的模块。 +8. Mark Phillips 的《[5 个 Ansible 运维任务][4]》(译文):这是另一个有关于如何使用 Ansible 来管理常见的系统操作任务的文章。这里描述了一系列可以取代命令行操作的 Tower(或 AWX)的案例。 +9. Chris Short 的《[Ansible 快速入门指南][2]》是个可以下载的 PDF 文档。它可以作为一本随时拿来翻阅的手册。这篇文章的开头有助于初学者入门。同时,还包括了一些其他的研究领域,比如模块测试、系统管理任务和针对 K8S 对象的管理。 +10. Mark Phillips 的《[Ansible 参考指南,带有 Ansible Tower 和 GitHub 的 CI/CD,等等][11]》:这是一篇每月进行总结更新的文章,充满了有趣的链接。话题包括了 Ansible 的基础内容、管理 Netapp 的 E 系列存储产品、调试、打补丁包和其他一些相关内容。文章中还包括了一些视频以及一些聚会的链接。请查看详情。 + +如果你也有一些你喜爱的 Ansible 文章,那请留言告诉我们吧。 -------------------------------------------------------------------------------- @@ -45,8 +47,8 @@ via: https://opensource.com/article/19/12/ansible-resources 作者:[James Farrell][a] 选题:[lujun9972][b] -译者:[译者ID](https://github.com/BrunoJu) -校对:[校对者ID](https://github.com/校对者ID) +译者:[BrunoJu](https://github.com/BrunoJu) +校对:[wxy](https://github.com/wxy) 本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出 @@ -55,11 +57,11 @@ via: https://opensource.com/article/19/12/ansible-resources [1]: https://opensource.com/sites/default/files/styles/image-full-size/public/lead-images/chaos_engineer_monster_scary_devops_gear_kubernetes.png?itok=GPYLvfVh (Gears above purple clouds) [2]: https://opensource.com/article/19/2/quickstart-guide-ansible [3]: https://opensource.com/article/19/9/must-know-ansible-modules -[4]: https://opensource.com/article/19/8/ops-tasks-ansible +[4]: https://linux.cn/article-11312-1.html [5]: https://opensource.com/article/19/4/ansible-procedures [6]: https://opensource.com/article/19/9/ansible-documentation-kids-laptops [7]: https://opensource.com/article/19/2/ansible-windows-admin -[8]: https://opensource.com/article/19/5/using-testinfra-ansible-verify-server-state +[8]: https://linux.cn/article-10943-1.html [9]: https://opensource.com/article/19/5/hardware-bootstrapping-ansible [10]: https://opensource.com/article/19/3/developing-ansible-modules [11]: https://opensource.com/article/19/7/ansible-news-edition-one From 9de7cd49aa0d08d79a379e08ffd875f2d72ff746 Mon Sep 17 00:00:00 2001 From: Xingyu Wang Date: Tue, 7 Jan 2020 23:13:18 +0800 Subject: [PATCH 03/20] PUB MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit @BrunoJu 本文首发地址:https://linux.cn/article-11760-1.html 您的 LCTT 专页地址: https://linux.cn/lctt/BrunoJu 请注册以领取 LCCN:https://lctt.linux.cn/ --- ... Ansible resources to accelerate your automation skills.md | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) rename {translated/tech => published}/20191231 10 Ansible resources to accelerate your automation skills.md (98%) diff --git a/translated/tech/20191231 10 Ansible resources to accelerate your automation skills.md b/published/20191231 10 Ansible resources to accelerate your automation skills.md similarity index 98% rename from translated/tech/20191231 10 Ansible resources to accelerate your automation skills.md rename to published/20191231 10 Ansible resources to accelerate your automation skills.md index f7805c35f0..00d31b98a2 100644 --- a/translated/tech/20191231 10 Ansible resources to accelerate your automation skills.md +++ b/published/20191231 10 Ansible resources to accelerate your automation skills.md @@ -1,8 +1,8 @@ [#]: collector: (lujun9972) [#]: translator: (BrunoJu) [#]: reviewer: (wxy) -[#]: publisher: ( ) -[#]: url: ( ) +[#]: publisher: (wxy) +[#]: url: (https://linux.cn/article-11760-1.html) [#]: subject: (10 Ansible resources to accelerate your automation skills) [#]: via: (https://opensource.com/article/19/12/ansible-resources) [#]: author: (James Farrell https://opensource.com/users/jamesf) From d817e328b23780246cb240b9f19e1f1fc4af841b Mon Sep 17 00:00:00 2001 From: Brooke Lau Date: Wed, 8 Jan 2020 00:20:46 +0800 Subject: [PATCH 04/20] translated --- ... Lessons learned from programming in Go.md | 56 +++++++++---------- 1 file changed, 28 insertions(+), 28 deletions(-) diff --git a/sources/tech/20191210 Lessons learned from programming in Go.md b/sources/tech/20191210 Lessons learned from programming in Go.md index 93e9af3398..682b6c798c 100644 --- a/sources/tech/20191210 Lessons learned from programming in Go.md +++ b/sources/tech/20191210 Lessons learned from programming in Go.md @@ -7,28 +7,28 @@ [#]: via: "https://opensource.com/article/19/12/go-common-pitfalls" [#]: author: "Eduardo Ferreira https://opensource.com/users/edufgf" -用 Go 编程受到的启发 +Go 编程中的经验教训 ====== 通过学习如何定位并发处理的陷阱来避免未来处理这些问题时的困境。 ![Goland gopher illustration][1] -在复杂的分布式系统进行任务处理时,你通常会需要进行并发的操作。[Mode.net][2] 公司系统每天要处理实时、快速和灵活的以毫秒为单位动态路由数据包的全球专用网络和数据,需要高度并发的系统。他们的动态路由是基于网络状态的,而这个过程需要考虑众多因素,我们只考虑关系链的监控。在我们的环境中,调用关系链监控可以是任何跟网络调用关系链有关的状态和当前属性(如连接延迟)。 +在复杂的分布式系统进行任务处理时,你通常会需要进行并发的操作。[Mode.net][2] 公司系统每天要处理实时、快速和灵活的以毫秒为单位动态路由数据包的全球专用网络和数据,需要高度并发的系统。这个动态路由是基于网络状态的,而这个过程需要考虑众多因素,我们只考虑关系链的监控。在我们的环境中,调用关系链监控可以是任何跟网络调用关系链有关的状态和当前属性(如链接延迟)。 -### 并发探测链接指标 +### 并发探测链接监控 -[H.A.L.O.][4] (Hop-by-Hop Adaptive Link-State Optimal Routing,译注:逐跳自适应链路状态最佳路由), 我们的动态路由算法部分依赖于链路度量来计算路由表。 这些指标由位于每个PoP(译注:存活节点)上的独立组件收集。PoP是表示我们的网络中单个路由实体的机器,通过链路连接并分布在我们的网络拓扑中的各个位置。某个组件使用网络数据包探测周围的机器,周围的机器回复数据包给前者。从接收到的探测包中可以获得链路延迟。由于每个 PoP 都有多个临近节点,所以这种探测任务实质上是并发的:我们需要实时测量每个临近连接点的延迟。我们不能串行地处理;为了计算这个指标,必须尽快处理每个探针。 +[H.A.L.O.][4] (Hop-by-Hop Adaptive Link-State Optimal Routing,译注:逐跳自适应链路状态最佳路由),我们的动态路由算法部分依赖于链路度量来计算路由表。 这些指标由位于每个PoP(译注:存活节点)上的独立组件收集。PoP是表示我们的网络中单个路由实体的机器,通过链路连接并分布在我们的网络拓扑中的各个位置。某个组件使用网络数据包探测周围的机器,周围的机器回复数据包给前者。从接收到的探测包中可以获得链路延迟。由于每个 PoP 都有不止一个临近节点,所以这种探测任务实质上是并发的:我们需要实时测量每个临近连接点的延迟。我们不能串行地处理;为了计算这个指标,必须尽快处理每个探针。 ![latency computation graph][6] ### 序列号和重置:一个记录场景 -我们的探测组件互相发送和接收数据包并依靠序列号进行数据包处理。旨在避免处理重复的包或顺序被打乱的包。我们的第一个实现依靠特殊的序列号0来重置序列号。这个数字仅在组件初始化时使用。主要的问题是我们只考虑了始终从 0 开始递增的序列号。组件重启后,包的顺序可能会重新排列,某个包的序列号可能会轻易地被替换成重置之前使用过的值。这意味着,直到排到重置之前用到的值之前,它后面的包都会被忽略掉。 +我们的探测组件互相发送和接收数据包并依靠序列号进行数据包处理。旨在避免处理重复的包或顺序被打乱的包。我们的第一个实现依靠特殊的序列号 0 来重置序列号。这个数字仅在组件初始化时使用。主要的问题是我们只考虑了始终从 0 开始递增的序列号。组件重启后,包的顺序可能会重新排列,某个包的序列号可能会轻易地被替换成重置之前使用过的值。这意味着,直到排到重置之前用到的值之前,它后面的包都会被忽略掉。 ### UDP 握手和有限状态机 这里的问题是重启前后的序列号是否一致。有几种方法可以解决这个问题,经过讨论,我们选择了实现一个带有清晰状态定义的三向交握协议。这个握手过程在初始化时通过链接建立 session。这样可以确保节点通过同一个 session 进行通信且使用了适当的序列号。 -为了正确实现这个过程,我们必须顶一个一个有清晰状态和过渡的有限状态机。这样我们就可以正确管理握手过程中的所有极端情况。 +为了正确实现这个过程,我们必须定义一个有清晰状态和过渡的有限状态机。这样我们就可以正确管理握手过程中的所有极端情况。 ![finite state machine diagram][7] @@ -43,7 +43,7 @@ session ID 由握手的初始化程序生成。一个完整的交换顺序如下 ### 处理状态超时 -Basically, at each state, you need to handle, at most, three types of events: link events, packet events, and timeout events. And those events show up concurrently, so here you have to handle concurrency properly.基本上,每种状态下你都需要处理最多三种类型的事件:链接事件、数据包事件和超时事件。这些事件会并发地出现,因此你必须正确处理并发。 +基本上,每种状态下你都需要处理最多三种类型的事件:链接事件、数据包事件和超时事件。这些事件会并发地出现,因此你必须正确处理并发。 * 链接事件包括连接和断开,连接时会初始化一个链接 session,断开时会断开一个已建立的 seesion。 * 数据包事件是控制数据包 **(SYN/SYN-ACK/ACK)** 或只是探测响应。 @@ -59,7 +59,7 @@ Basically, at each state, you need to handle, at most, three types of events: li ![gophers hacking together][9] -gophers hacking together +gopher 们聚众狂欢 首先,你可以设计两个分别表示我们的 **Session** 和 **Timeout Handlers** 的结构体。 @@ -147,19 +147,19 @@ func synCallback(session Session) {   ### 数据竞争和引用 -你的解决方案已经有了。可以通过检查计时器到期后超时回调是否执行来进行一个简单的测试。为此,注册一个超时,在 *duration* 时间内 sleep,然后检查是否执行的回调的处理。执行这个测试后,最好取消预定的超时时间(因为它会重新调度),这样才不会在下次测试时产生副作用。 +你的解决方案已经有了。可以通过检查计时器到期后超时回调是否执行来进行一个简单的测试。为此,注册一个超时,在 *duration* 时间内 sleep,然后检查是否执行了回调的处理。执行这个测试后,最好取消预定的超时时间(因为它会重新调度),这样才不会在下次测试时产生副作用。 令人惊讶的是,这个简单的测试发现了这个解决方案中的一个 bug。使用 cancel 方法来取消超时并没有正确处理。以下顺序的事件会导致数据资源竞争: 1. 你有一个已调度的超时 handler。 2. 线程 1: -a) 你接收到一个控制数据包,现在你要取消已注册的超时并切换到下一个 session 状态(如 发送 **SYN** 后接收到一个 **SYN-ACK**) -b) 你调用了 **timeout.Cancel()**,这个函数调用了 **timer.Stop()**。(请注意,Golang 计时器的 stop 不会终止一个已过期的计时器。) +a)你接收到一个控制数据包,现在你要取消已注册的超时并切换到下一个 session 状态(如 发送 **SYN** 后接收到一个 **SYN-ACK**) +b)你调用了 **timeout.Cancel()**,这个函数调用了 **timer.Stop()**。(请注意,Golang 计时器的 stop 不会终止一个已过期的计时器。) 3. 线程 2: -a) 在调用 cancel 之前,计时器已过期,回调即将执行。 -b) 执行回调,它调度一次新的超时并更新全局 map。 +a)在调用 cancel 之前,计时器已过期,回调即将执行。 +b)执行回调,它调度一次新的超时并更新全局 map。 4. 线程 1: -a) 切换到新的 session 状态并注册新的超时,更新全局 map。 +a)切换到新的 session 状态并注册新的超时,更新全局 map。 @@ -187,9 +187,9 @@ func (timeout* TimeoutHandler) Register() {   你可以使用取消 channel,而不必依赖不能阻止到期的计时器执行的 golang 函数 **timer.Stop()**。 -这是一个略有不同的方法。现在你可以不用再通过回调进行递归地重新调度;而是注册一个死循环,这个循环接收到取消信号或超时事件是终止。 +这是一个略有不同的方法。现在你可以不用再通过回调进行递归地重新调度;而是注册一个死循环,这个循环接收到取消信号或超时事件时终止。 -新的 **Register()** 产生一个新的 go 协程,这个协程在在超时后执行你的回调,并在前一个超时执行后调度新的超时。返回给调用发一个取消 channel,用来控制循环的终止。 +新的 **Register()** 产生一个新的 go 协程,这个协程在在超时后执行你的回调,并在前一个超时执行后调度新的超时。返回给调用方一个取消 channel,用来控制循环的终止。 ```go @@ -254,11 +254,11 @@ func (timeout* TimeoutHandler) Cancel() {   ![gophers on a wire, talking][11] -还有人吗? +有人在听吗? 已获得 Trevor Forrey 授权。 -这里的解决方案是创建 channel 时指定大小至少为 1,这样向 channel 发送数据就不会阻塞,也显式地使发送变成非阻塞的,避免了并发调用。这样可以确保取消操作只发送一次,并且不会则色后续的取消调用。 +这里的解决方案是创建 channel 时指定大小至少为 1,这样向 channel 发送数据就不会阻塞,也显式地使发送变成非阻塞的,避免了并发调用。这样可以确保取消操作只发送一次,并且不会阻塞后续的取消调用。 ```go @@ -281,29 +281,29 @@ func (timeout* TimeoutHandler) Cancel() {   #### 在非同步的情况下更新共享数据 -This seems like an obvious one, but it’s actually hard to spot if your concurrent updates happen in different locations. The result is data race, where multiple updates to the same data can cause update loss, due to one update overriding another. In our case, we were updating the scheduled timeout reference on the same shared map. (Interestingly, if Go detects a concurrent read/write on the same Map object, it throws a fatal error —you can try to run Go’s [data race detector][12]). This eventually results in losing a timeout reference and making it impossible to cancel that given timeout. Always remember to use locks when they are needed. +这似乎是个很明显的问题,但如果并发更新发生在不同的位置,就很难发现。结果就是数据竞争,由于一个更新会覆盖另一个,因此对同一数据的多次更新中会有某些更新丢失。在我们的案例中,我们是在同时更新同一个共享 map 里的调度超时引用。有趣的是,(如果 Go 检测到在同一个 map 对象上的并发读写,会抛出 fatal 错误 — 你可以尝试下运行 Go 的[数据竞争检测器](https://golang.org/doc/articles/race_detector.html))。这最终会导致丢失超时引用,且无法取消给定的超时。当有必要时,永远不要忘记使用锁。 ![gopher assembly line][13] -don’t forget to synchronize gophers’ work +不要忘记同步 gopher 们的工作 -#### Missing condition checks +#### 缺少条件检查 -Condition checks are needed in situations where you can’t rely only on the lock exclusivity. Our situation is a bit different, but the core idea is the same as [condition variables][14]. Imagine a classic situation where you have one producer and multiple consumers working with a shared queue. A producer can add one item to the queue and wake up all consumers. The wake-up call means that some data is available at the queue, and because the queue is shared, access must be synchronized through a lock. Every consumer has a chance to grab the lock; however, you still need to check if there are items in the queue. A condition check is needed because you don’t know the queue status by the time you grab the lock. +在不能仅依赖锁的独占性的情况下,就需要进行条件检查。我们遇到的场景稍微有点不一样,但是核心思想跟[条件变量][14]是一样的。假设有个经典的一个生产者和多个消费者使用一个共享队列的场景,生产者可以将一个元素添加到队列并唤醒所有消费者。这个唤醒调用意味着队列中的数据是可访问的,并且由于队列是共享的,消费者必须通过锁来进行同步访问。每个消费者都可能拿到锁;然而,你仍然需要检查队列中是否有元素。因为在你拿到锁的瞬间并不知道队列的状态,所以还是需要进行条件检查。 -In our example, the timeout handler got a ‘wake up’ call from a timer expiration, but it still needed to check if a cancel signal was sent to it before it could proceed with the callback execution. +在我们的例子中,超时 handler 收到了计时器到期时发出的「唤醒」调用,但是它仍需要检查是否已向其发送了取消信号,然后才能继续执行回调。 ![gopher boot camp][15] -condition checks might be needed if you wake up multiple gophers +如果你要唤醒多个 gopher,可能就需要进行条件检查 -#### Deadlocks +#### 死锁 -This happens when one thread is stuck, waiting indefinitely for a signal to wake up, but this signal will never arrive. Those can completely kill your application by halting your entire program execution. +当一个线程被卡住,无限期地等待一个唤醒信号,但是这个信号永远不会到达时,就会发生这种情况。死锁可以通过让你的整个程序 halt 来彻底杀死你的应用。 -In our case, this happened due to multiple send calls to a non-buffered and blocking channel. This meant that the send call would only return after a receive is done on the same channel. Our timeout thread loop was promptly receiving signals on the cancellation channel; however, after the first signal is received, it would break off the loop and never read from that channel again. The remaining callers are stuck forever. To avoid this situation, you need to carefully think through your code, handle blocking calls with care, and guarantee that thread starvation doesn’t happen. The fix in our example was to make the cancellation calls non-blocking—we didn’t need a blocking call for our needs. +在我们的案例中,这种情况的发生是由于多次发送请求到一个非缓冲且阻塞的 channel。这意味着向 channel 发送数据只有在从这个 channel 接收完数据后才能 return。我们的超时线程循环迅速从取消 channel 接收信号;然而,在接收到第一个信号后,它将跳出循环,并且再也不会从这个 channel 读取数据。其他的调用会一直被卡住。为避免这种情况,你需要仔细检查代码,谨慎处理阻塞调用,并确保不会发生线程饥饿。我们例子中的解决方法是使取消调用成为非阻塞调用 — 我们不需要阻塞调用。 -------------------------------------------------------------------------------- @@ -311,7 +311,7 @@ via: https://opensource.com/article/19/12/go-common-pitfalls 作者:[Eduardo Ferreira][a] 选题:[lujun9972][b] -译者:[译者ID](https://github.com/译者ID) +译者:[lxbwolf](https://github.com/lxbwolf) 校对:[校对者ID](https://github.com/校对者ID) 本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出 From e37b92517847b91b7e86bcbb64f0897f44f1f50a Mon Sep 17 00:00:00 2001 From: Brooke Lau Date: Wed, 8 Jan 2020 00:23:40 +0800 Subject: [PATCH 05/20] modify md --- ... Lessons learned from programming in Go.md | 47 ++++++------------- 1 file changed, 15 insertions(+), 32 deletions(-) diff --git a/sources/tech/20191210 Lessons learned from programming in Go.md b/sources/tech/20191210 Lessons learned from programming in Go.md index 682b6c798c..0c17e142a8 100644 --- a/sources/tech/20191210 Lessons learned from programming in Go.md +++ b/sources/tech/20191210 Lessons learned from programming in Go.md @@ -1,11 +1,11 @@ -[#]: collector: "lujun9972" -[#]: translator: "lxbwolf" -[#]: reviewer: " " -[#]: publisher: " " -[#]: url: " " -[#]: subject: "Lessons learned from programming in Go" -[#]: via: "https://opensource.com/article/19/12/go-common-pitfalls" -[#]: author: "Eduardo Ferreira https://opensource.com/users/edufgf" +[#]: collector: (lujun9972) +[#]: translator: (lxbwolf) +[#]: reviewer: ( ) +[#]: publisher: ( ) +[#]: url: ( ) +[#]: subject: (Lessons learned from programming in Go) +[#]: via: (https://opensource.com/article/19/12/go-common-pitfalls) +[#]: author: (Eduardo Ferreira https://opensource.com/users/edufgf) Go 编程中的经验教训 ====== @@ -39,8 +39,6 @@ session ID 由握手的初始化程序生成。一个完整的交换顺序如下 3. sender 接收到 **SYN-ACK (ID)** _并发送一个 **ACK (ID)**_。它还发送一个从序列号 0 开始的数据包。 4. receiver 检查最后接收到的 **ID**,如果 ID 匹配,_则接受 **ACK (ID)**_。它还开始接受序列号为 0 的数据包。 - - ### 处理状态超时 基本上,每种状态下你都需要处理最多三种类型的事件:链接事件、数据包事件和超时事件。这些事件会并发地出现,因此你必须正确处理并发。 @@ -49,8 +47,6 @@ session ID 由握手的初始化程序生成。一个完整的交换顺序如下 * 数据包事件是控制数据包 **(SYN/SYN-ACK/ACK)** 或只是探测响应。 * 超时事件在当前 session 状态的预定超时时间到期后触发。 - - 这里面临的最主要的问题是如何处理并发超时到期和其他事件。这里很容易陷入死锁和资源竞争的陷阱。 ### 第一种方法 @@ -63,7 +59,6 @@ gopher 们聚众狂欢 首先,你可以设计两个分别表示我们的 **Session** 和 **Timeout Handlers** 的结构体。 - ```go type Session struct {     State SessionState   @@ -85,14 +80,12 @@ type TimeoutHandler struct {   每一个临近连接点的 session 都包含一个保存调度 `TimeoutHandler` 的全局 map。 - ``` `SessionTimeout map[Session]*TimeoutHandler` ``` 下面方法注册和取消超时: - ```go // schedules the timeout callback function.   func (timeout* TimeoutHandler) Register() {   @@ -111,7 +104,6 @@ func (timeout* TimeoutHandler) Cancel() {   你可以使用类似下面的方法来创建和存储超时: - ```go func CreateTimeoutHandler(callback func(Session), session Session, duration int) *TimeoutHandler {     if sessionTimeout[session] == nil {   @@ -130,7 +122,6 @@ func CreateTimeoutHandler(callback func(Session), session Session, duration int) 为此,你可以让回调函数重新调度一次超时: - ```go func synCallback(session Session) {     sendSynPacket(session) @@ -161,15 +152,12 @@ b)执行回调,它调度一次新的超时并更新全局 map。 4. 线程 1: a)切换到新的 session 状态并注册新的超时,更新全局 map。 - - 两个线程同时更新超时 map。最终结果是你无法取消注册的超时,然后你也会丢失对线程 2 重新调度的超时的引用。这导致 handler 在一段时间内持续执行和重新调度,出现非预期行为。 ### 锁也解决不了问题 使用锁也不能完全解决问题。如果你在处理所有事件和执行回调之前加锁,它仍然不能阻止一个过期的回调运行: - ```go func (timeout* TimeoutHandler) Register() {     timeout.timer = time.AfterFunc(time.Duration(timeout.duration) * time._Second_, func() {   @@ -191,7 +179,6 @@ func (timeout* TimeoutHandler) Register() {   新的 **Register()** 产生一个新的 go 协程,这个协程在在超时后执行你的回调,并在前一个超时执行后调度新的超时。返回给调用方一个取消 channel,用来控制循环的终止。 - ```go func (timeout *TimeoutHandler) Register() chan struct{} {     cancelChan := make(chan struct{})   @@ -225,7 +212,6 @@ func (timeout* TimeoutHandler) Cancel() {   这里的解决方案是,在拿到锁之后,检查一下超时范围内的取消 channel。 - ```go   case _ = <- time.AfterFunc(time.Duration(timeout.duration) * time.Second):       func () {   @@ -260,7 +246,6 @@ func (timeout* TimeoutHandler) Cancel() {   这里的解决方案是创建 channel 时指定大小至少为 1,这样向 channel 发送数据就不会阻塞,也显式地使发送变成非阻塞的,避免了并发调用。这样可以确保取消操作只发送一次,并且不会阻塞后续的取消调用。 - ```go func (timeout* TimeoutHandler) Cancel() {     if timeout.cancelChan == nil {   @@ -283,8 +268,6 @@ func (timeout* TimeoutHandler) Cancel() {   这似乎是个很明显的问题,但如果并发更新发生在不同的位置,就很难发现。结果就是数据竞争,由于一个更新会覆盖另一个,因此对同一数据的多次更新中会有某些更新丢失。在我们的案例中,我们是在同时更新同一个共享 map 里的调度超时引用。有趣的是,(如果 Go 检测到在同一个 map 对象上的并发读写,会抛出 fatal 错误 — 你可以尝试下运行 Go 的[数据竞争检测器](https://golang.org/doc/articles/race_detector.html))。这最终会导致丢失超时引用,且无法取消给定的超时。当有必要时,永远不要忘记使用锁。 - - ![gopher assembly line][13] 不要忘记同步 gopher 们的工作 @@ -318,18 +301,18 @@ via: https://opensource.com/article/19/12/go-common-pitfalls [a]: https://opensource.com/users/edufgf [b]: https://github.com/lujun9972 -[1]: https://opensource.com/sites/default/files/styles/image-full-size/public/lead-images/go-golang.png?itok=OAW9BXny "Goland gopher illustration" +[1]: https://opensource.com/sites/default/files/styles/image-full-size/public/lead-images/go-golang.png?itok=OAW9BXny (Goland gopher illustration) [2]: http://mode.net [3]: https://en.wikipedia.org/wiki/Metrics_%28networking%29 [4]: https://people.ece.cornell.edu/atang/pub/15/HALO_ToN.pdf [5]: https://en.wikipedia.org/wiki/Point_of_presence -[6]: https://opensource.com/sites/default/files/uploads/image2_0_3.png "latency computation graph" -[7]: https://opensource.com/sites/default/files/uploads/image3_0.png "finite state machine diagram" +[6]: https://opensource.com/sites/default/files/uploads/image2_0_3.png (latency computation graph) +[7]: https://opensource.com/sites/default/files/uploads/image3_0.png (finite state machine diagram) [8]: https://golang.org/ -[9]: https://opensource.com/sites/default/files/uploads/image4.png "gophers hacking together" +[9]: https://opensource.com/sites/default/files/uploads/image4.png (gophers hacking together) [10]: https://en.wikipedia.org/wiki/Deadlock -[11]: https://opensource.com/sites/default/files/uploads/image5_0_0.jpg "gophers on a wire, talking" +[11]: https://opensource.com/sites/default/files/uploads/image5_0_0.jpg (gophers on a wire, talking) [12]: https://golang.org/doc/articles/race_detector.html -[13]: https://opensource.com/sites/default/files/uploads/image6.jpeg "gopher assembly line" +[13]: https://opensource.com/sites/default/files/uploads/image6.jpeg (gopher assembly line) [14]: https://en.wikipedia.org/wiki/Monitor_%28synchronization%29#Condition_variables -[15]: https://opensource.com/sites/default/files/uploads/image7.png "gopher boot camp" +[15]: https://opensource.com/sites/default/files/uploads/image7.png (gopher boot camp) From 327565767baa82996ecd556f11f27a6907183d03 Mon Sep 17 00:00:00 2001 From: Brooke Lau Date: Wed, 8 Jan 2020 00:24:07 +0800 Subject: [PATCH 06/20] move file --- .../tech/20191210 Lessons learned from programming in Go.md | 0 1 file changed, 0 insertions(+), 0 deletions(-) rename {sources => translated}/tech/20191210 Lessons learned from programming in Go.md (100%) diff --git a/sources/tech/20191210 Lessons learned from programming in Go.md b/translated/tech/20191210 Lessons learned from programming in Go.md similarity index 100% rename from sources/tech/20191210 Lessons learned from programming in Go.md rename to translated/tech/20191210 Lessons learned from programming in Go.md From 90dcfb8cbe828bff61e12427e2aebf571a89ca9c Mon Sep 17 00:00:00 2001 From: DarkSun Date: Wed, 8 Jan 2020 00:54:36 +0800 Subject: [PATCH 07/20] =?UTF-8?q?=E9=80=89=E9=A2=98:=2020200107=205=20ways?= =?UTF-8?q?=20to=20improve=20your=20Bash=20scripts?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit sources/tech/20200107 5 ways to improve your Bash scripts.md --- ...107 5 ways to improve your Bash scripts.md | 177 ++++++++++++++++++ 1 file changed, 177 insertions(+) create mode 100644 sources/tech/20200107 5 ways to improve your Bash scripts.md diff --git a/sources/tech/20200107 5 ways to improve your Bash scripts.md b/sources/tech/20200107 5 ways to improve your Bash scripts.md new file mode 100644 index 0000000000..ae4dd4fcf7 --- /dev/null +++ b/sources/tech/20200107 5 ways to improve your Bash scripts.md @@ -0,0 +1,177 @@ +[#]: collector: (lujun9972) +[#]: translator: ( ) +[#]: reviewer: ( ) +[#]: publisher: ( ) +[#]: url: ( ) +[#]: subject: (5 ways to improve your Bash scripts) +[#]: via: (https://opensource.com/article/20/1/improve-bash-scripts) +[#]: author: (Alan Formy-Duval https://opensource.com/users/alanfdoss) + +5 ways to improve your Bash scripts +====== +Find out how Bash can help you tackle the most challenging tasks. +![A person working.][1] + +A system admin often writes Bash scripts, some short and some quite lengthy, to accomplish various tasks. + +Have you ever looked at an installation script provided by a software vendor? They often add a lot of functions and logic in order to ensure that the installation works properly and doesn’t result in damage to the customer’s system. Over the years, I’ve amassed a collection of various techniques for enhancing my Bash scripts, and I’d like to share some of them in hopes they can help others. Here is a collection of small scripts created to illustrate these simple examples. + +### Starting out + +When I was starting out, my Bash scripts were nothing more than a series of commands, usually meant to save time with standard shell operations like deploying web content. One such task was extracting static content into the home directory of an Apache web server. My script went something like this: + + +``` +cp january_schedule.tar.gz /usr/apache/home/calendar/ +cd /usr/apache/home/calendar/ +tar zvxf january_schedule.tar.gz +``` + +While this saved me some time and typing, it certainly was not a very interesting or useful script in the long term. Over time, I learned other ways to use Bash scripts to accomplish more challenging tasks, such as creating software packages, installing software, or backing up a file server. + +### 1\. The conditional statement + +Just as with so many other programming languages, the conditional has been a powerful and common feature. A conditional is what enables logic to be performed by a computer program. Most of my examples are based on conditional logic. + +The basic conditional uses an "if" statement. This allows us to test for some condition that we can then use to manipulate how a script performs. For instance, we can check for the existence of a Java bin directory, which would indicate that Java is installed. If found, the executable path can be updated with the location to enable calls by Java applications. + + +``` +if [ -d "$JAVA_HOME/bin" ] ; then +    PATH="$JAVA_HOME/bin:$PATH" +``` + +### 2\. Limit execution + +You might want to limit a script to only be run by a specific user. Although Linux has standard permissions for users and groups, as well as SELinux for enabling this type of protection, you could choose to place logic within a script. Perhaps you want to be sure that only the owner of a particular web application can run its startup script. You could even use code to limit a script to the root user. Linux has a couple of environment variables that we can test in this logic. One is **$USER**, which provides the username. Another is **$UID**, which provides the user’s identification number (UID) and, in the case of a script, the UID of the executing user. + +#### User + +The first example shows how I could limit a script to the user jboss1 in a multi-hosting environment with several application server instances. The conditional "if" statement essentially asks, "Is the executing user not jboss1?" When the condition is found to be true, the first echo statement is called, followed by the **exit 1,** which terminates the script. + + +``` +if [ "$USER" != 'jboss1' ]; then +     echo "Sorry, this script must be run as JBOSS1!" +     exit 1 +fi +echo "continue script" +``` + +#### Root + +This next example script ensures that only the root user can execute it. Because the UID for root is 0, we can use the **-gt** option in the conditional if statement to prohibit all UIDs greater than zero. + + +``` +if [ "$UID" -gt 0 ]; then +     echo "Sorry, this script must be run as ROOT!" +     exit 1 +fi +echo "continue script" +``` + +### 3\. Use arguments + +Just like any executable program, Bash scripts can take arguments as input. Below are a few examples. But first, you should understand that good programming means that we don’t just write applications that do what we want; we must write applications that _can’t_ do what we _don’t_ want. I like to ensure that a script doesn’t do anything destructive in the case where there is no argument. Therefore, this is the first check that y. The condition checks the number of arguments, **$#**, for a value of zero and terminates the script if true. + + +``` +if [ $# -eq 0 ]; then +    echo "No arguments provided" +    exit 1 +fi +echo "arguments found: $#" +``` + +#### Multiple arguments + +You can pass more than one argument to a script. The internal variables that the script uses to reference each argument are simply incremented, such as **$1**, **$2**, **$3**, and so on. I’ll just expand my example above with the following line to echo the first three arguments. Obviously, additional logic will be needed for proper argument handling based on the total number. This example is simple for the sake of demonstration. + + +``` +`echo $1 $2 $3` +``` + +While we’re discussing these argument variables, you might have wondered, "Did he skip zero?" + +Well, yes, I did, but I have a great reason! There is indeed a **$0** variable, and it is very useful. Its value is simply the name of the script being executed. + + +``` +`echo $0` +``` + +An important reason to reference the name of the script during execution is to generate a log file that includes the script’s name in its own name. The simplest form might just be an echo statement. + + +``` +`echo test >> $0.log` +``` + +However, you will probably want to add a bit more code to ensure that the log is written to a location with the name and information that you find helpful to your use case. + +### 4\. User input + +Another useful feature to use in a script is its ability to accept input during execution. The simplest is to offer the user some input. + + +``` +echo "enter a word please:" + read word + echo $word +``` + +This also allows you to provide choices to the user. + + +``` +read -p "Install Software ?? [Y/n]: " answ + if [ "$answ" == 'n' ]; then +   exit 1 + fi +   echo "Installation starting..." +``` + +### 5\. Exit on failure + +Some years ago, I wrote a script for installing the latest version of the Java Development Kit (JDK) on my computer. The script extracts the JDK archive to a specific directory, updates a symbolic link, and uses the alternatives utility to make the system aware of the new version. If the extraction of the JDK archive failed, continuing could break Java system-wide. So, I wanted the script to abort in such a situation. I don’t want the script to make the next set of system changes unless the archive was successfully extracted. The following is an excerpt from that script: + + +``` +tar kxzmf jdk-8u221-linux-x64.tar.gz -C /jdk --checkpoint=.500; ec=$? +if [ $ec -ne 0 ]; then +     echo "Installation failed - exiting." +     exit 1 +fi +``` + +A quick way for you to demonstrate the usage of the **$?** variable is with this short one-liner: + + +``` +`ls T; ec=$?; echo $ec` +``` + +First, run **touch T** followed by this command. The value of **ec** will be 0. Then, delete **T**, **rm T**, and repeat the command. The value of **ec** will now be 2 because ls reports an error condition since **T** was not found. + +You can take advantage of this error reporting to include logic, as I have above, to control the behavior of your scripts. + +### Takeaway + +We might assume that we need to employ languages, such as Python, C, or Java, for higher functionality, but that’s not necessarily true. The Bash scripting language is very powerful. There is a lot to learn to maximize its usefulness. I hope these few examples will shed some light on the potential of coding with Bash. + +-------------------------------------------------------------------------------- + +via: https://opensource.com/article/20/1/improve-bash-scripts + +作者:[Alan Formy-Duval][a] +选题:[lujun9972][b] +译者:[译者ID](https://github.com/译者ID) +校对:[校对者ID](https://github.com/校对者ID) + +本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出 + +[a]: https://opensource.com/users/alanfdoss +[b]: https://github.com/lujun9972 +[1]: https://opensource.com/sites/default/files/styles/image-full-size/public/lead-images/rh_003784_02_os.comcareers_os_rh2x.png?itok=jbRfXinl (A person working.) From 909fa992fb18d393ca1463f0f2d57ec01e9d0c51 Mon Sep 17 00:00:00 2001 From: DarkSun Date: Wed, 8 Jan 2020 00:55:09 +0800 Subject: [PATCH 08/20] =?UTF-8?q?=E9=80=89=E9=A2=98:=2020200107=20How=20pi?= =?UTF-8?q?wheels=20will=20save=20Raspberry=20Pi=20users=20time=20in=20202?= =?UTF-8?q?0?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit sources/tech/20200107 How piwheels will save Raspberry Pi users time in 2020.md --- ...ll save Raspberry Pi users time in 2020.md | 127 ++++++++++++++++++ 1 file changed, 127 insertions(+) create mode 100644 sources/tech/20200107 How piwheels will save Raspberry Pi users time in 2020.md diff --git a/sources/tech/20200107 How piwheels will save Raspberry Pi users time in 2020.md b/sources/tech/20200107 How piwheels will save Raspberry Pi users time in 2020.md new file mode 100644 index 0000000000..aa43d04f90 --- /dev/null +++ b/sources/tech/20200107 How piwheels will save Raspberry Pi users time in 2020.md @@ -0,0 +1,127 @@ +[#]: collector: (lujun9972) +[#]: translator: ( ) +[#]: reviewer: ( ) +[#]: publisher: ( ) +[#]: url: ( ) +[#]: subject: (How piwheels will save Raspberry Pi users time in 2020) +[#]: via: (https://opensource.com/article/20/1/piwheels) +[#]: author: (Ben Nuttall https://opensource.com/users/bennuttall) + +How piwheels will save Raspberry Pi users time in 2020 +====== +By making pre-compiled Python packages for Raspberry Pi available, the +piwheels project saves users significant time and effort. +![rainbow colors on pinwheels in the sun][1] + +Piwheels automates building Python wheels (pre-compiled Python packages) for all of the projects on [PyPI][2], the Python Package Index, using Raspberry Pi hardware to ensure compatibility. This means that when a Raspberry Pi user wants to install a Python library using **pip**, they get a ready-made compiled version that's guaranteed to work on the Raspberry Pi. This makes it much easier for Raspberry Pi users to dive in and get started with their projects. + +![Piwheels logo][3] + +When I wrote [_piwheels: Speedy Python package installation for the Raspberry Pi_][4] in October 2018, the piwheels project was in its first year and already proving its purpose of saving Raspberry Pi users considerable time and effort. But the project, which makes pre-compiled Python packages available for Raspberry Pi, has come a long way in its second year. + +![Raspberry Pi 4][5] + +### How it works + +[Raspbian][6], the primary OS for Raspberry Pi, comes pre-configured to use piwheels, so users don't need to do anything special to get access to the wheels. + +The configuration file (at **/etc/pip.conf**) tells pip to use [piwheels.org][7] as an _additional index_, so pip looks at PyPI first, then piwheels. The Piwheels website is hosted on a Raspberry Pi 3, and all the wheels built by the project are hosted on that Pi. It serves over 1 million packages per month—not bad for a $35 computer! + +In addition to the main Raspberry Pi that serves the website, the piwheels project uses seven other Pis to build the packages. Some run Raspbian Jessie, building wheels for Python 3.4, some run Raspbian Stretch for Python 3.5, and some run Raspbian Buster for Python 3.7. The project doesn't generally support other Python versions. There's also a "proper server"—a virtual machine running the Postgres database. Since the Pi 3 has just 1GB of RAM, the (very large) database doesn't run well on it, so we moved it to a VM. The Pi 4 with 4GB RAM would probably be suitable, so we may move to this in the future. + +The Pis are all on an IPv6-only network in a "Pi Cloud"—a brilliant service provided by Cambridge-based hosting company [Mythic Beasts][8]. + +![Mythic Beasts hosting service][9] + +### Downloads and trends + +Every time a wheel file is downloaded, it is logged in the database. This provides insight into what packages are most popular and what Python versions and operating systems people are using. We don't have much information from the user agent, but because the architecture of Pi 1/Zero shows as "armv6" and Pi 2/3/4 show as "armv7," we can tell them apart. + +As of mid-December 2019, over 14 million packages have been downloaded from piwheels, with nearly 9 million in 2019 alone. + +The 10 most popular packages since the project's inception are: + + 1. [pycparser][10] (821,060 downloads) + 2. [PyYAML][11] (366,979) + 3. [numpy][12] (354,531) + 4. [cffi][13] (336,982) + 5. [MarkupSafe][14] (318,878) + 6. [future][15] (282,349) + 7. [aiohttp][16] (277,046) + 8. [cryptography][17] (276,167) + 9. [home-assistant-frontend][18] (266,667) + 10. [multidict][19] (256,185) + + + +Note that many pure-Python packages, such as [urllib3][20], are provided as wheels on PyPI; because these are compatible across platforms, they're not usually downloaded from piwheels because PyPI takes precedence. + +We also see trends in things like which Python versions are used over time. This shows the quick takeover of Python 3.7 from 3.5 when Raspbian Buster was released: + +![Data from piwheels on Python versions used over time][21] + +You can see more trends in our [stats blog posts][22]. + +### Time saved + +Every package build is logged in the database, and every download is also stored. Cross-referencing downloads with build duration shows how much time has been saved. One example is numpy—the latest version took about 11 minutes to build. + +So far, piwheels has saved users a total of over 165 years of build time. At the current usage rate, piwheels saves _over 200 days per day_. + +As well as saving build time, having pre-compiled wheels also means people don't have to install various development tools to build packages. Some packages require other apt packages for them to access shared libraries. Figuring out which ones you need can be a pain, so we made that step easier, too. First, we figured out the process and [documented it on our blog][23]. Then we added this logic to the build process so that when a wheel is built, its dependencies are automatically calculated and added to the package's project page: + +![numpy dependencies][24] + +### What next for piwheels? + +We launched project pages (e.g., [numpy][25]) this year, which are a really useful way to let people look up information about a project in a human-readable way. They also make it easier for people to report issues, such as if a project is missing from piwheels or they have an issue with a package they've downloaded. + +In early 2020, we're planning to roll out some upgrades to piwheels that will enable a new JSON API, so you can automatically check which versions are available, look up dependencies for a project, and lots more. + +The next Debian/Raspbian upgrade won't happen until mid-2021, so we won't start building wheels for any new Python versions until then. + +You can read more about piwheels on the project's [blog][26], where I'll be publishing a 2019 roundup early in 2020. You can also follow [@piwheels][27] on Twitter, where you'll see daily and monthly stats along with any milestones reached. + +Of course, piwheels is an open source project, and you can see the entire project [source code on GitHub][28]. + +-------------------------------------------------------------------------------- + +via: https://opensource.com/article/20/1/piwheels + +作者:[Ben Nuttall][a] +选题:[lujun9972][b] +译者:[译者ID](https://github.com/译者ID) +校对:[校对者ID](https://github.com/校对者ID) + +本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出 + +[a]: https://opensource.com/users/bennuttall +[b]: https://github.com/lujun9972 +[1]: https://opensource.com/sites/default/files/styles/image-full-size/public/lead-images/rainbow-pinwheel-piwheel-diversity-inclusion.png?itok=di41Wd3V (rainbow colors on pinwheels in the sun) +[2]: https://pypi.org/ +[3]: https://opensource.com/sites/default/files/uploads/piwheels.png (Piwheels logo) +[4]: https://opensource.com/article/18/10/piwheels-python-raspberrypi +[5]: https://opensource.com/sites/default/files/uploads/raspberry-pi-4_0.jpg (Raspberry Pi 4) +[6]: https://www.raspberrypi.org/downloads/raspbian/ +[7]: http://piwheels.org +[8]: https://www.mythic-beasts.com/order/rpi +[9]: https://opensource.com/sites/default/files/uploads/pi-cloud.png (Mythic Beasts hosting service) +[10]: https://www.piwheels.org/project/pycparser +[11]: https://www.piwheels.org/project/PyYAML +[12]: https://www.piwheels.org/project/numpy +[13]: https://www.piwheels.org/project/cffi +[14]: https://www.piwheels.org/project/MarkupSafe +[15]: https://www.piwheels.org/project/future +[16]: https://www.piwheels.org/project/aiohttp +[17]: https://www.piwheels.org/project/cryptography +[18]: https://www.piwheels.org/project/home-assistant-frontend +[19]: https://www.piwheels.org/project/multidict +[20]: https://piwheels.org/project/urllib3/ +[21]: https://opensource.com/sites/default/files/uploads/pyvers2019.png (Data from piwheels on Python versions used over time) +[22]: https://blog.piwheels.org/piwheels-stats-for-2019/ +[23]: https://blog.piwheels.org/how-to-work-out-the-missing-dependencies-for-a-python-package/ +[24]: https://opensource.com/sites/default/files/uploads/numpy-deps.png (numpy dependencies) +[25]: https://www.piwheels.org/project/numpy/ +[26]: https://blog.piwheels.org/ +[27]: https://twitter.com/piwheels +[28]: https://github.com/piwheels/ From e1088cdf2c4aee98fff42e8232b1227161721579 Mon Sep 17 00:00:00 2001 From: MonkeyDEcho <863626156@qq.com> Date: Wed, 8 Jan 2020 01:20:21 +0800 Subject: [PATCH 09/20] =?UTF-8?q?=E7=94=B3=E8=AF=B7=E8=AF=A5=E6=96=87?= =?UTF-8?q?=E7=AB=A0=EF=BC=8C=E7=BF=BB=E8=AF=9150%?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../tech/20181105 5 Minimal Web Browsers for Linux.md | 10 +++++++++- 1 file changed, 9 insertions(+), 1 deletion(-) diff --git a/sources/tech/20181105 5 Minimal Web Browsers for Linux.md b/sources/tech/20181105 5 Minimal Web Browsers for Linux.md index 1fbc18aeae..34c0c1e18e 100644 --- a/sources/tech/20181105 5 Minimal Web Browsers for Linux.md +++ b/sources/tech/20181105 5 Minimal Web Browsers for Linux.md @@ -1,5 +1,5 @@ [#]: collector: (lujun9972) -[#]: translator: ( ) +[#]: translator: (MonkeyDEcho ) [#]: reviewer: ( ) [#]: publisher: ( ) [#]: subject: (5 Minimal Web Browsers for Linux) @@ -9,19 +9,27 @@ 5 Minimal Web Browsers for Linux ====== +linux上的五种微型浏览器 +====== + ![](https://www.linux.com/sites/lcom/files/styles/rendered_file/public/minimal.jpg?itok=ifA0Y3pV) There are so many reasons to enjoy the Linux desktop. One reason I often state up front is the almost unlimited number of choices to be found at almost every conceivable level. From how you interact with the operating system (via a desktop interface), to how daemons run, to what tools you use, you have a multitude of options. +有太多理由去选择使用linux系统。很重要的一个理由是,我们可以按照我们自己的想法去选择想要的。从操作系统的交互方式(桌面系统)到守护系统的运行方式,在到使用的工具,你用更多的选择。 The same thing goes for web browsers. You can use anything from open source favorites, such as [Firefox][1] and [Chromium][2], or closed sourced industry darlings like [Vivaldi][3] and [Chrome][4]. Those options are full-fledged browsers with every possible bell and whistle you’ll ever need. For some, these feature-rich browsers are perfect for everyday needs. +web浏览器也是如此。你可以使用开源的[火狐][1],[Chromium][2];或者未开源的[Vivaldi][3],[Chrome][4]。这些功能强大的浏览器有你需要的各种功能。对于某些人,这些功能完备的浏览器是日常必需的。 There are those, however, who prefer using a web browser without all the frills. In fact, there are many reasons why you might prefer a minimal browser over a standard browser. For some, it’s about browser security, while others look at a web browser as a single-function tool (as opposed to a one-stop shop application). Still others might be running low-powered machines that cannot handle the requirements of, say, Firefox or Chrome. Regardless of the reason, Linux has you covered. +但是,有些人更喜欢没有冗余功能的纯粹的浏览器。实际上,有很多原因导致你会选择微型的浏览器而不选择上述功能完备的浏览器。对于某些人来说,与浏览器的安全有关;而有些人则将浏览器当作一种简单的工具(而不是一站式商店应用程序);还有一些可能运行在低功率的计算机上,这些计算机无法满足火狐,chrome浏览器的运行要求。无论出于何种原因,在linux系统上都可以满足你的要求。 Let’s take a look at five of the minimal browsers that can be installed on Linux. I’ll be demonstrating these browsers on the Elementary OS platform, but each of these browsers are available to nearly every distribution in the known Linuxverse. Let’s dive in. +让我们看一下可以在linux上安装运行的五种微型浏览器。我将在 Elementary 的操作系统平台上演示这些浏览器,在已知的linux发型版中几乎每个版本都可以使用这些浏览器。让我们一起来看一下吧! ### GNOME Web GNOME Web (codename Epiphany, which means [“a usually sudden manifestation or perception of the essential nature or meaning of something”][5]) is the default web browser for Elementary OS, but it can be installed from the standard repositories. (Note, however, that the recommended installation of Epiphany is via Flatpak or Snap). If you choose to install via the standard package manager, issue a command such as sudo apt-get install epiphany-browser -y for successful installation. +GNOME web (Epiphany 含义:[顿悟][5])是Elementary系统默认的web浏览器,也可以从标准存储库中安装。(注意,建议通过使用 Flatpak 或者 Snap 工具安装),如果你想选择标准软件包管理器进行安装,请执行 ```sudo apt-get install epiphany-browser -y``` 命令成功安装。 Epiphany uses the WebKit rendering engine, which is the same engine used in Apple’s Safari browser. Couple that rendering engine with the fact that Epiphany has very little in terms of bloat to get in the way, you will enjoy very fast page-rendering speeds. Epiphany development follows strict adherence to the following guidelines: From 42f1a83728463efea3411631f34a871d0e1305c6 Mon Sep 17 00:00:00 2001 From: MonkeyDEcho <863626156@qq.com> Date: Wed, 8 Jan 2020 01:23:58 +0800 Subject: [PATCH 10/20] fix --- sources/tech/20181105 5 Minimal Web Browsers for Linux.md | 7 ------- 1 file changed, 7 deletions(-) diff --git a/sources/tech/20181105 5 Minimal Web Browsers for Linux.md b/sources/tech/20181105 5 Minimal Web Browsers for Linux.md index 34c0c1e18e..e3ee66f40a 100644 --- a/sources/tech/20181105 5 Minimal Web Browsers for Linux.md +++ b/sources/tech/20181105 5 Minimal Web Browsers for Linux.md @@ -9,27 +9,20 @@ 5 Minimal Web Browsers for Linux ====== -linux上的五种微型浏览器 -====== ![](https://www.linux.com/sites/lcom/files/styles/rendered_file/public/minimal.jpg?itok=ifA0Y3pV) There are so many reasons to enjoy the Linux desktop. One reason I often state up front is the almost unlimited number of choices to be found at almost every conceivable level. From how you interact with the operating system (via a desktop interface), to how daemons run, to what tools you use, you have a multitude of options. -有太多理由去选择使用linux系统。很重要的一个理由是,我们可以按照我们自己的想法去选择想要的。从操作系统的交互方式(桌面系统)到守护系统的运行方式,在到使用的工具,你用更多的选择。 The same thing goes for web browsers. You can use anything from open source favorites, such as [Firefox][1] and [Chromium][2], or closed sourced industry darlings like [Vivaldi][3] and [Chrome][4]. Those options are full-fledged browsers with every possible bell and whistle you’ll ever need. For some, these feature-rich browsers are perfect for everyday needs. -web浏览器也是如此。你可以使用开源的[火狐][1],[Chromium][2];或者未开源的[Vivaldi][3],[Chrome][4]。这些功能强大的浏览器有你需要的各种功能。对于某些人,这些功能完备的浏览器是日常必需的。 There are those, however, who prefer using a web browser without all the frills. In fact, there are many reasons why you might prefer a minimal browser over a standard browser. For some, it’s about browser security, while others look at a web browser as a single-function tool (as opposed to a one-stop shop application). Still others might be running low-powered machines that cannot handle the requirements of, say, Firefox or Chrome. Regardless of the reason, Linux has you covered. -但是,有些人更喜欢没有冗余功能的纯粹的浏览器。实际上,有很多原因导致你会选择微型的浏览器而不选择上述功能完备的浏览器。对于某些人来说,与浏览器的安全有关;而有些人则将浏览器当作一种简单的工具(而不是一站式商店应用程序);还有一些可能运行在低功率的计算机上,这些计算机无法满足火狐,chrome浏览器的运行要求。无论出于何种原因,在linux系统上都可以满足你的要求。 Let’s take a look at five of the minimal browsers that can be installed on Linux. I’ll be demonstrating these browsers on the Elementary OS platform, but each of these browsers are available to nearly every distribution in the known Linuxverse. Let’s dive in. -让我们看一下可以在linux上安装运行的五种微型浏览器。我将在 Elementary 的操作系统平台上演示这些浏览器,在已知的linux发型版中几乎每个版本都可以使用这些浏览器。让我们一起来看一下吧! ### GNOME Web GNOME Web (codename Epiphany, which means [“a usually sudden manifestation or perception of the essential nature or meaning of something”][5]) is the default web browser for Elementary OS, but it can be installed from the standard repositories. (Note, however, that the recommended installation of Epiphany is via Flatpak or Snap). If you choose to install via the standard package manager, issue a command such as sudo apt-get install epiphany-browser -y for successful installation. -GNOME web (Epiphany 含义:[顿悟][5])是Elementary系统默认的web浏览器,也可以从标准存储库中安装。(注意,建议通过使用 Flatpak 或者 Snap 工具安装),如果你想选择标准软件包管理器进行安装,请执行 ```sudo apt-get install epiphany-browser -y``` 命令成功安装。 Epiphany uses the WebKit rendering engine, which is the same engine used in Apple’s Safari browser. Couple that rendering engine with the fact that Epiphany has very little in terms of bloat to get in the way, you will enjoy very fast page-rendering speeds. Epiphany development follows strict adherence to the following guidelines: From ad719218b29a617feb3d0fe931e90c9976409350 Mon Sep 17 00:00:00 2001 From: MonkeyDEcho <863626156@qq.com> Date: Wed, 8 Jan 2020 01:31:32 +0800 Subject: [PATCH 11/20] Update 20181105 5 Minimal Web Browsers for Linux.md --- sources/tech/20181105 5 Minimal Web Browsers for Linux.md | 1 - 1 file changed, 1 deletion(-) diff --git a/sources/tech/20181105 5 Minimal Web Browsers for Linux.md b/sources/tech/20181105 5 Minimal Web Browsers for Linux.md index e3ee66f40a..70d75f74e7 100644 --- a/sources/tech/20181105 5 Minimal Web Browsers for Linux.md +++ b/sources/tech/20181105 5 Minimal Web Browsers for Linux.md @@ -9,7 +9,6 @@ 5 Minimal Web Browsers for Linux ====== - ![](https://www.linux.com/sites/lcom/files/styles/rendered_file/public/minimal.jpg?itok=ifA0Y3pV) There are so many reasons to enjoy the Linux desktop. One reason I often state up front is the almost unlimited number of choices to be found at almost every conceivable level. From how you interact with the operating system (via a desktop interface), to how daemons run, to what tools you use, you have a multitude of options. From 1cd57105714a3362b0c1acc0e74a58da5c0c5c79 Mon Sep 17 00:00:00 2001 From: MonkeyDEcho <863626156@qq.com> Date: Wed, 8 Jan 2020 01:35:03 +0800 Subject: [PATCH 12/20] fix --- sources/tech/20181105 5 Minimal Web Browsers for Linux.md | 8 ++++++++ 1 file changed, 8 insertions(+) diff --git a/sources/tech/20181105 5 Minimal Web Browsers for Linux.md b/sources/tech/20181105 5 Minimal Web Browsers for Linux.md index 70d75f74e7..34c0c1e18e 100644 --- a/sources/tech/20181105 5 Minimal Web Browsers for Linux.md +++ b/sources/tech/20181105 5 Minimal Web Browsers for Linux.md @@ -9,19 +9,27 @@ 5 Minimal Web Browsers for Linux ====== +linux上的五种微型浏览器 +====== + ![](https://www.linux.com/sites/lcom/files/styles/rendered_file/public/minimal.jpg?itok=ifA0Y3pV) There are so many reasons to enjoy the Linux desktop. One reason I often state up front is the almost unlimited number of choices to be found at almost every conceivable level. From how you interact with the operating system (via a desktop interface), to how daemons run, to what tools you use, you have a multitude of options. +有太多理由去选择使用linux系统。很重要的一个理由是,我们可以按照我们自己的想法去选择想要的。从操作系统的交互方式(桌面系统)到守护系统的运行方式,在到使用的工具,你用更多的选择。 The same thing goes for web browsers. You can use anything from open source favorites, such as [Firefox][1] and [Chromium][2], or closed sourced industry darlings like [Vivaldi][3] and [Chrome][4]. Those options are full-fledged browsers with every possible bell and whistle you’ll ever need. For some, these feature-rich browsers are perfect for everyday needs. +web浏览器也是如此。你可以使用开源的[火狐][1],[Chromium][2];或者未开源的[Vivaldi][3],[Chrome][4]。这些功能强大的浏览器有你需要的各种功能。对于某些人,这些功能完备的浏览器是日常必需的。 There are those, however, who prefer using a web browser without all the frills. In fact, there are many reasons why you might prefer a minimal browser over a standard browser. For some, it’s about browser security, while others look at a web browser as a single-function tool (as opposed to a one-stop shop application). Still others might be running low-powered machines that cannot handle the requirements of, say, Firefox or Chrome. Regardless of the reason, Linux has you covered. +但是,有些人更喜欢没有冗余功能的纯粹的浏览器。实际上,有很多原因导致你会选择微型的浏览器而不选择上述功能完备的浏览器。对于某些人来说,与浏览器的安全有关;而有些人则将浏览器当作一种简单的工具(而不是一站式商店应用程序);还有一些可能运行在低功率的计算机上,这些计算机无法满足火狐,chrome浏览器的运行要求。无论出于何种原因,在linux系统上都可以满足你的要求。 Let’s take a look at five of the minimal browsers that can be installed on Linux. I’ll be demonstrating these browsers on the Elementary OS platform, but each of these browsers are available to nearly every distribution in the known Linuxverse. Let’s dive in. +让我们看一下可以在linux上安装运行的五种微型浏览器。我将在 Elementary 的操作系统平台上演示这些浏览器,在已知的linux发型版中几乎每个版本都可以使用这些浏览器。让我们一起来看一下吧! ### GNOME Web GNOME Web (codename Epiphany, which means [“a usually sudden manifestation or perception of the essential nature or meaning of something”][5]) is the default web browser for Elementary OS, but it can be installed from the standard repositories. (Note, however, that the recommended installation of Epiphany is via Flatpak or Snap). If you choose to install via the standard package manager, issue a command such as sudo apt-get install epiphany-browser -y for successful installation. +GNOME web (Epiphany 含义:[顿悟][5])是Elementary系统默认的web浏览器,也可以从标准存储库中安装。(注意,建议通过使用 Flatpak 或者 Snap 工具安装),如果你想选择标准软件包管理器进行安装,请执行 ```sudo apt-get install epiphany-browser -y``` 命令成功安装。 Epiphany uses the WebKit rendering engine, which is the same engine used in Apple’s Safari browser. Couple that rendering engine with the fact that Epiphany has very little in terms of bloat to get in the way, you will enjoy very fast page-rendering speeds. Epiphany development follows strict adherence to the following guidelines: From 0e6d1adb756702f050b674b34988029175449188 Mon Sep 17 00:00:00 2001 From: geekpi Date: Wed, 8 Jan 2020 08:43:15 +0800 Subject: [PATCH 13/20] translated --- ...VLC More Awesome With These Simple Tips.md | 131 ----------------- ...VLC More Awesome With These Simple Tips.md | 132 ++++++++++++++++++ 2 files changed, 132 insertions(+), 131 deletions(-) delete mode 100644 sources/tech/20191214 Make VLC More Awesome With These Simple Tips.md create mode 100644 translated/tech/20191214 Make VLC More Awesome With These Simple Tips.md diff --git a/sources/tech/20191214 Make VLC More Awesome With These Simple Tips.md b/sources/tech/20191214 Make VLC More Awesome With These Simple Tips.md deleted file mode 100644 index a4dd3b7bd5..0000000000 --- a/sources/tech/20191214 Make VLC More Awesome With These Simple Tips.md +++ /dev/null @@ -1,131 +0,0 @@ -[#]: collector: (lujun9972) -[#]: translator: (geekpi) -[#]: reviewer: ( ) -[#]: publisher: ( ) -[#]: url: ( ) -[#]: subject: (Make VLC More Awesome With These Simple Tips) -[#]: via: (https://itsfoss.com/simple-vlc-tips/) -[#]: author: (Ankush Das https://itsfoss.com/author/ankush/) - -Make VLC More Awesome With These Simple Tips -====== - -[VLC][1] is one of the [best open source video players][2], if not the best. What most people don’t know about it is that it is a lot more than just a video player. - -You can do a lot of complex tasks like broadcasting live videos, capturing devices etc. Just open its menu and you’ll see how many options it has. - -It’s FOSS has a detailed tutorial discussing some of the [pro VLC tricks][3] but those are way too complicated for normal users. - -This is why I am writing another article to show you some of the simple tips that you can use with VLC. - -### Do more with VLC with these simple tips - -Let’s see what can you do with VLC other than just playing a video file. - -#### 1\. Watch YouTube videos with VLC - -![][4] - -If you do not want to watch the annoying advertisements on [YouTube][5] or simply want a distraction-free experience for watching a YouTube video, you can use VLC. - -Yes, it is very easy to stream a YouTube video on VLC. - -Simply launch the VLC player, head to the Media settings and click on “**Open Network Stream**” or **CTRL + N** as a shortcut to that. - -![][6] - -Next, you just have to paste the URL of the video that you want to watch. There are some options to tweak – usually, you should not bother using them. But, if you are curious you can click on the “**Advanced options**” to explore. - -You can also add subtitles to the YouTube videos this way. However, an easier way to [watch YouTube or any online video with subtitles is using Penguin subtitle player][7]. - -#### 2\. Convert videos to different formats - -![][8] - -You can [use ffmpeg to convert videos in Linux command line][9]. You can also use a graphical tool like [HandBrake to convert video formats][10]. - -But if you do not want a separate app to transcode videos, you can use VLC media player to get the job done. - -To do that, just head on to the Media option on VLC and then click on “**Convert/Save**” or press CTRL + R as a shortcut to get there while you have VLC media player active. - -Next, you will need to either import the video from your computer/disk or paste the URL of the video that you want to save/convert. - -Whatever your input source is – just hit the “**Convert/Save**” button after selecting the file. - -Now, you will find another window that gives you the option to change the “**Profile**” from the settings. Click on it and choose a format that you’d like the video to be converted to (and saved). - -You can also change the storage path for the converted file by setting the destination folder at the bottom of the screen before converting it. - -#### 3\. Record Audio/Video From Source - -![Vlc Advanced Controls][11] - -Do you want to record the audio/video you’re playing on VLC Media Player? - -If yes, there’s an easy solution to that. Simply navigate your way through **View->click on “Advanced Controls”**. - -Once you do that, you should observe new buttons (including a red record button in your VLC player). - -#### 4\. Download subtitles automatically - -![][12] - -Yes, you can [automatically download subtitles with VLC][13]. You do not even have to look for it on a separate website. You just have to navigate your way to **View->VLSub**. - -By default, it is deactivated, so when you click on the option it gets activated and lets you search/download the subtitles you wanted. - -[VLC also lets you synchronize the subtitles][14] with simple keyboard shortcuts. - -#### 5\. Take A Snapshot - -![][15] - -With VLC, you can get some screenshots/images of the video while watching it. - -You just need to right-click on the player while the video is playing/paused, you will notice a bunch of options now, navigate through **Video->Take Snapshot**. - -If you have an old version installed, you might observe the snapshot option right after performing a right-click. - -#### Bonus Tip: Add Audio/Video Effects to a video - -From the menu, go to the “**Tools**” option. Now, click on “**Effects and Filters**” or simply press **CTRL + E** from the VLC player window to open up the option. - -Here, you can observe audio effects and video effects that you can add to your video. You may not be able to see all the changes in real-time, so you will have to tweak it and save it in order to see what happens. - -![][16] - -I’ll suggest keeping a backup of the original video before you modify the video. - -#### What’s your favorite VLC tip? - -I shared some of my favourite VLC tips. Do you know some cool tip that you use regularly with VLC? Why not share it with us? I may add it to the list here. - --------------------------------------------------------------------------------- - -via: https://itsfoss.com/simple-vlc-tips/ - -作者:[Ankush Das][a] -选题:[lujun9972][b] -译者:[译者ID](https://github.com/译者ID) -校对:[校对者ID](https://github.com/校对者ID) - -本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出 - -[a]: https://itsfoss.com/author/ankush/ -[b]: https://github.com/lujun9972 -[1]: https://www.videolan.org/ -[2]: https://itsfoss.com/video-players-linux/ -[3]: https://itsfoss.com/vlc-pro-tricks-linux/ -[4]: https://i1.wp.com/itsfoss.com/wp-content/uploads/2019/12/youtube-video-stream.jpg?ssl=1 -[5]: https://www.youtube.com/ -[6]: https://i1.wp.com/itsfoss.com/wp-content/uploads/2019/12/youtube-video-play.jpg?ssl=1 -[7]: https://itsfoss.com/penguin-subtitle-player/ -[8]: https://i0.wp.com/itsfoss.com/wp-content/uploads/2019/12/vlc-video-convert.jpg?ssl=1 -[9]: https://itsfoss.com/ffmpeg/ -[10]: https://itsfoss.com/handbrake/ -[11]: https://i0.wp.com/itsfoss.com/wp-content/uploads/2019/12/vlc-advanced-controls.png?ssl=1 -[12]: https://i1.wp.com/itsfoss.com/wp-content/uploads/2019/12/vlc-subtitles-automatic.png?ssl=1 -[13]: https://itsfoss.com/download-subtitles-automatically-vlc-media-player-ubuntu/ -[14]: https://itsfoss.com/how-to-synchronize-subtitles-with-movie-quick-tip/ -[15]: https://i1.wp.com/itsfoss.com/wp-content/uploads/2019/12/vlc-snapshot.png?ssl=1 -[16]: https://i1.wp.com/itsfoss.com/wp-content/uploads/2019/12/vlc-effects-screenshot.jpg?ssl=1 diff --git a/translated/tech/20191214 Make VLC More Awesome With These Simple Tips.md b/translated/tech/20191214 Make VLC More Awesome With These Simple Tips.md new file mode 100644 index 0000000000..7363cb7992 --- /dev/null +++ b/translated/tech/20191214 Make VLC More Awesome With These Simple Tips.md @@ -0,0 +1,132 @@ +[#]: collector: (lujun9972) +[#]: translator: (geekpi) +[#]: reviewer: ( ) +[#]: publisher: ( ) +[#]: url: ( ) +[#]: subject: (Make VLC More Awesome With These Simple Tips) +[#]: via: (https://itsfoss.com/simple-vlc-tips/) +[#]: author: (Ankush Das https://itsfoss.com/author/ankush/) + +这些简单的技巧使 VLC 更加出色 +====== + +如果 [VLC][1] 不是最好的播放器,那它是[最好的开源视频播放器][2]之一。大多数人不知道的是,它不仅仅是视频播放器。 + +你可以进行许多复杂的任务,如直播视频、捕捉设备等。只需打开菜单,你就可以看到它有多少选项。 + +它的 FOSS 页面有一个详细的教程,讨论一些[专业的 VLC 技巧][3],但这些对于普通用户太复杂。 + +这就是为什么我写另一篇文章的原因,来向你展示一些可以在 VLC 中使用的简单技巧。 + +### 使用这些简单技巧让 VLC 做更多事 + +Let’s see what can you do with VLC other than just playing a video file. +让我们看看除了播放视频文件之外,你还可以使用 VLC 做什么。 + +#### 1\. 使用 VLC 观看 YouTube 视频 + +![][4] + +如果你不想在 [YouTube][5] 上观看令人讨厌的广告,或者只想体验没有打扰地观看 YouTube 视频,你可以使用 VLC。 + +是的,在 VLC 上流式传输 YouTube 视频是非常容易的。 + +只需启动 VLC 播放器,前往媒体设置,然后单击 ”**Open Network Stream**“ 或使用快捷方式 **CTRL + N**。 + +![][6] + +接下来,你只需要粘贴要观看的视频的 URL。有一些选项可以调整,但通常你无需担心这些。如果你好奇,你可以点击 ”**Advanced options**“ 来探索。 + +你还可以通过这种方式向 YouTube 视频添加字幕。然而,[一个更简单的带字幕观看 Youtube 视频的办法是使用 Penguin 字幕播放器][7]。 + +#### 2\. 将视频转换为不同格式 + +![][8] + +你可以[在 Linux 命令行使用 ffmpeg 转换视频][9]。你还可以使用图形工具,如 [HandBrake 转换视频格式][10]。 + +但是,如果你不想用一个单独的应用转码视频,你可以使用 VLC 播放器来完成该工作。 + +为此,只需点击 VLC 上的媒体选项,然后单击 **Convert/Save**,或者在 VLC 播放器处于活动状态时按下快捷键 CTRL + R。 + +接下来,你需要从计算机/硬盘或者 URL 导入你想保存/转换的的视频。 + +不管什么来源,只需选择文件后点击 ”**Convert/Save**“ 按钮 + +你现在会看到另外一个窗口给你更改 ”**Profile**“ 设置。点击并选择你想转换的格式(并保存)。 + +你还可以在转换之前通过在屏幕底部设置目标文件夹来更改转换文件的存储路径。 + +#### 3\. 从源录制音频/视频 + +![Vlc Advanced Controls][11] + +你是否想在 VLC 播放器中录制正在播放的音频/视频? + +如果是的话,有一个简单的解决方案。只需**通过 View,然后点击 ”Advanced Controls“**。 + +完成后,你会看到一个新按钮(包括 VLC 播放器中的红色录制按钮)。 + +#### 4\. 自动下载字幕 + +![][12] + +是的,你可以[使用 VLC 自动下载字幕][13]。你甚至不必在单独的网站上查找字幕。你只需点击 **View->VLSub**。 + +默认情况下,它是禁用的,因此当你单击该选项时,它会被激活,并允许你搜索/下载想要的字幕。 + +[VLC 还能让你使用简单的键盘快捷键同步字幕][14] + +#### 5\. 截图 + +![][15] + +你可以在观看视频时使用 VLC 获取一些视频的截图/图像。 + +你只需在视频播放/暂停时右击播放器,你会看到一组选项,点击 **Video->Take Snapshot**。 + +如果安装了旧版本,你可能在右键时看到截图选项。 + +#### 额外技巧:给视频添加音频/视频效果 + +在菜单中,进入 ”**Tools**“ 选项。单击 ”**Effects and Filters**“,或者在 VLC 播放器窗口中按 **CTRL + E** 打开选项。 + +好了,你可以观察你给视频添加的音频和视频效果了。你也许无法实时看到效果,因此你需要调整并保存来看发生了什么。 + +![][16] + +我建议在修改视频之前保存一份原始视频备份。 + +#### 你最喜欢的 VLC 技巧是什么? + +我分享了一些我最喜欢的 VLC 技巧。你知道什么你经常使用的很酷的 VLC 技巧吗?为什么不和我们分享呢?我可以把它添加到列表中。 + +-------------------------------------------------------------------------------- + +via: https://itsfoss.com/simple-vlc-tips/ + +作者:[Ankush Das][a] +选题:[lujun9972][b] +译者:[geekpi](https://github.com/geekpi) +校对:[校对者ID](https://github.com/校对者ID) + +本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出 + +[a]: https://itsfoss.com/author/ankush/ +[b]: https://github.com/lujun9972 +[1]: https://www.videolan.org/ +[2]: https://itsfoss.com/video-players-linux/ +[3]: https://itsfoss.com/vlc-pro-tricks-linux/ +[4]: https://i1.wp.com/itsfoss.com/wp-content/uploads/2019/12/youtube-video-stream.jpg?ssl=1 +[5]: https://www.youtube.com/ +[6]: https://i1.wp.com/itsfoss.com/wp-content/uploads/2019/12/youtube-video-play.jpg?ssl=1 +[7]: https://itsfoss.com/penguin-subtitle-player/ +[8]: https://i0.wp.com/itsfoss.com/wp-content/uploads/2019/12/vlc-video-convert.jpg?ssl=1 +[9]: https://itsfoss.com/ffmpeg/ +[10]: https://itsfoss.com/handbrake/ +[11]: https://i0.wp.com/itsfoss.com/wp-content/uploads/2019/12/vlc-advanced-controls.png?ssl=1 +[12]: https://i1.wp.com/itsfoss.com/wp-content/uploads/2019/12/vlc-subtitles-automatic.png?ssl=1 +[13]: https://itsfoss.com/download-subtitles-automatically-vlc-media-player-ubuntu/ +[14]: https://itsfoss.com/how-to-synchronize-subtitles-with-movie-quick-tip/ +[15]: https://i1.wp.com/itsfoss.com/wp-content/uploads/2019/12/vlc-snapshot.png?ssl=1 +[16]: https://i1.wp.com/itsfoss.com/wp-content/uploads/2019/12/vlc-effects-screenshot.jpg?ssl=1 From 4a0eed84886d7440734a2900198135556a798d83 Mon Sep 17 00:00:00 2001 From: geekpi Date: Wed, 8 Jan 2020 08:47:19 +0800 Subject: [PATCH 14/20] translating --- .../tech/20200105 PaperWM- tiled window management for GNOME.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/sources/tech/20200105 PaperWM- tiled window management for GNOME.md b/sources/tech/20200105 PaperWM- tiled window management for GNOME.md index 0948014fba..fa99f6c352 100644 --- a/sources/tech/20200105 PaperWM- tiled window management for GNOME.md +++ b/sources/tech/20200105 PaperWM- tiled window management for GNOME.md @@ -1,5 +1,5 @@ [#]: collector: (lujun9972) -[#]: translator: ( ) +[#]: translator: (geekpi) [#]: reviewer: ( ) [#]: publisher: ( ) [#]: url: ( ) From f213a05ee5b45f7a12ca8772157b6ccc91268dfe Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E4=BB=98=E5=B3=A5?= <24203166+fuzheng1998@users.noreply.github.com> Date: Wed, 8 Jan 2020 10:29:19 +0800 Subject: [PATCH 15/20] Update 20200104 Shocking- EA is Permanently Banning Linux Gamers on Battlefield V.md --- ...- EA is Permanently Banning Linux Gamers on Battlefield V.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/sources/news/20200104 Shocking- EA is Permanently Banning Linux Gamers on Battlefield V.md b/sources/news/20200104 Shocking- EA is Permanently Banning Linux Gamers on Battlefield V.md index b0d8bb3874..b0818e5fe8 100644 --- a/sources/news/20200104 Shocking- EA is Permanently Banning Linux Gamers on Battlefield V.md +++ b/sources/news/20200104 Shocking- EA is Permanently Banning Linux Gamers on Battlefield V.md @@ -1,5 +1,5 @@ [#]: collector: (lujun9972) -[#]: translator: ( ) +[#]: translator: (fuzheng1998) [#]: reviewer: ( ) [#]: publisher: ( ) [#]: url: ( ) From d25e4d31152b2f43164afb5e92fdac207274fb37 Mon Sep 17 00:00:00 2001 From: BrunoJu Date: Wed, 8 Jan 2020 11:59:16 +0800 Subject: [PATCH 16/20] Update 20200107 Kali Linux Will No Longer Have The Default Root User.md --- ...0107 Kali Linux Will No Longer Have The Default Root User.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/sources/tech/20200107 Kali Linux Will No Longer Have The Default Root User.md b/sources/tech/20200107 Kali Linux Will No Longer Have The Default Root User.md index 2ffcd8f715..0f087529b9 100644 --- a/sources/tech/20200107 Kali Linux Will No Longer Have The Default Root User.md +++ b/sources/tech/20200107 Kali Linux Will No Longer Have The Default Root User.md @@ -1,5 +1,5 @@ [#]: collector: (lujun9972) -[#]: translator: ( ) +[#]: translator: (BrunoJu) [#]: reviewer: ( ) [#]: publisher: ( ) [#]: url: ( ) From 7114530f7fa58b5a473b638039ad6bcf7db09bca Mon Sep 17 00:00:00 2001 From: Xingyu Wang Date: Wed, 8 Jan 2020 13:55:50 +0800 Subject: [PATCH 17/20] PRF @wxy --- ...191130 7 maker gifts for kids and teens.md | 37 ++++++++++--------- 1 file changed, 19 insertions(+), 18 deletions(-) diff --git a/translated/tech/20191130 7 maker gifts for kids and teens.md b/translated/tech/20191130 7 maker gifts for kids and teens.md index 59bc3fc837..002e97e442 100644 --- a/translated/tech/20191130 7 maker gifts for kids and teens.md +++ b/translated/tech/20191130 7 maker gifts for kids and teens.md @@ -1,15 +1,16 @@ [#]: collector: (lujun9972) [#]: translator: (wxy) -[#]: reviewer: ( ) +[#]: reviewer: (wxy) [#]: publisher: ( ) [#]: url: ( ) [#]: subject: (7 maker gifts for kids and teens) [#]: via: (https://opensource.com/article/19/11/maker-gifts-kids) [#]: author: (Jess Weichler https://opensource.com/users/cyanide-cupcake) -7 个给儿童和少年的创客礼物 +给儿童和青少年的 7 件创客礼物 ====== -> 这份礼物指南可给婴儿、儿童、青少年及年龄更大的人们带来创造和创新能力,使你轻松完成节日礼物的采购。 + +> 这份礼物指南使你轻松完成节日礼物的采购,它们可给婴儿、儿童、青少年及年龄更大的人们带来创造和创新能力。 ![Gift box opens with colors coming out][1] @@ -19,13 +20,13 @@ ![Hummingbird Robotics Kit][2] -**年龄:**8 岁 - 成人 +**年龄:**8 岁 - 成人 -**这是什么:**[蜂鸟机器人套件][3]是一套完整的机器人套件,带有微控制器、电机、LED 和传感器。机器人的大脑具有特殊的端口,小手可以轻松地将其连接到机器人的组件上。蜂鸟套件并没有身体,鼓励用户自己创建一个。 +**这是什么:**[蜂鸟机器人套件][3]是一套完整的机器人套件,带有微控制器、电机、LED 和传感器。机器人的大脑具有特殊的端口,小手可以轻松地将其连接到机器人的组件上。蜂鸟套件并没有身体,而是鼓励用户自己创建一个。 **为什么我喜欢它:**蜂鸟可以使用多种编程语言 —— 从可视化编程(BirdBlox、MakeCode、Snap)到代码编程(Python 和 Java)—— 可以随着用户编码技能的提高而可扩展。所有组件均与你在电子商店中找到的组件完全相同,没有像其他机器人套件那样被塑料所遮盖。这使机器人的内部工作不再神秘,并在你需要时易于采购更多零件。 -由于没有固定组合项目,因此蜂鸟是发挥创造力的完美机器人。 +由于没有固定组装项目,因此蜂鸟是发挥创造力的完美机器人。 蜂鸟具有开源的软件和固件。它适用于 Linux、Windows、Mac、Chromebook、Android 和 iOS。 @@ -37,7 +38,7 @@ **年龄:** 6岁 - 成人 -**这是什么:** [Makey Makey 经典版][5]可将任何导电物体(从棉花糖到朋友)变成计算机钥匙。 +**这是什么:** [Makey Makey 经典版][5]可将任何导电物体(从棉花糖到你的朋友)变成计算机钥匙。 你可以使用鳄鱼夹将 Makey Makey 连接到你选择的导电物体上。然后,通过同时触摸两个导电物体来闭合接地和任何触发键之间的电路。Makey Makey 是一种安全的方法,可以安全地在家中探索电力,同时创造与计算机进行交互的有趣方式。 @@ -51,11 +52,11 @@ **年龄:** 10 岁 - 成人 -**这是什么:** Arduino 是随同电子套件购买的微控制器,也可以单独购买,它们具有多种版本,尽管我最喜欢[Arduino Uno][7]。可以根据需要从任何电子商店购买其他组件,例如 LED、电机和传感器。 +**这是什么:** Arduino 是随同电子套件购买的微控制器,也可以单独购买,它们具有多种版本,而我最喜欢 [Arduino Uno][7]。你可以根据需要从任何电子商店购买其他组件,例如 LED、电机和传感器。 **为什么我喜欢它:** Arduino Uno 的文档很完善,因此创客们很容易在线上找到教程。Arduino 可以实现从简单到复杂的各种电子项目。Arduino 具有开源的固件和硬件。它适用于 Linux、Mac 和 Windows。 -**费用:**主板的起价为 22.00 美元。总成本取决于项目和技能水平。 +**费用:** 主板的起价为 22.00 美元。总成本取决于项目和技能水平。 ### DIY 创客套件 @@ -63,7 +64,7 @@ **年龄**:8 岁 - 成人 -**这是什么:**当今许多创客、发明家和程序员都是从碰巧修补附加的东西开始的。你可以快速前往最近的电子产品商店,为家里的年轻人创建一套出色的创客工具包。这是我的创客工具包中的内容: +**这是什么:**当今许多创客、发明家和程序员都是从鼓捣碰巧出现在身边东西开始的。你可以快速前往最近的电子产品商店,为家里的年轻人创建一套出色的创客工具包。这是我的创客工具包中的内容: * 护目镜 * 锤子 @@ -74,7 +75,7 @@ * LED * 压电蜂鸣器 * 马达 -* 带引线的AA电池组 +* 带引线的 AA 电池组 * 剪线钳 * 纸板 * 美纹纸胶带 @@ -85,7 +86,7 @@ * 拉链 * 钩子 * 一个很酷的工具盒,用来存放所有东西 -   + **我为什么喜欢它:**还记得小时候,你把父母带回家的空纸箱变成了宇宙飞船、房屋或超级计算机吗?这就是为大孩子们准备的 DIY 创客工具包。 原始的组件使孩子们可以尝试并运用他们的想象力。DIY 创客工具包可以完全针对接收者定制。可以放入一些接受这份礼品的人可能从未想到过用之发挥创意的某些组件,例如为下水道提供一些 LED 或木工结构。 @@ -98,7 +99,7 @@ **年龄:** 8 个月至 5 岁 -**这是什么:**启发式游戏篮充满了由天然、无毒材料制成的有趣物品,可供婴幼儿使用其五种感官进行探索。这是一种开放式、自娱自乐的游戏。其想法是,成年人将监督(但不指导)儿童使用篮子及其物品半小时,然后将篮子拿走下一次再玩。 +**这是什么:**启发式游戏篮充满了由天然、无毒材料制成的有趣物品,可供婴幼儿使用其五种感官进行探索。这是一种开放式、自娱自乐的游戏。其想法是,成年人将监督(但不指导)儿童使用篮子及其物品半小时,然后将篮子拿走,等下一次再玩。 创建带有常见家用物品的可爱游戏篮很容易。尝试包括质地、声音、气味、形状和重量各不相同的物品。这里有一些想法可以帮助您入门。 @@ -107,7 +108,7 @@ * 金属打蛋器和汤匙 * 板刷 * 海绵 -* 小鸡蛋纸箱 +* 小型鸡蛋纸箱 * 纸板管 * 小擀面杖 * 带纹理的毛巾 @@ -120,7 +121,7 @@ **我为什么喜欢它:**游戏篮非常适合感官发育,并可以帮助幼儿提出问题和探索周围的世界。这是培养创客思维方式的重要组成部分! -很容易获得适合这个游戏篮的物品。你可能已经在家中或附近的二手店里有很多有趣的物品。幼儿使用游戏篮的方式与婴儿不同。随着孩子开始模仿成人生活并通过他们的游戏讲故事,这些物品将随孩子一起成长。 +很容易获得适合这个游戏篮的物品。你可能已经在家中或附近的二手商店里找到了很多有趣的物品。幼儿使用游戏篮的方式与婴儿不同。随着孩子们开始模仿成人生活并通过他们的游戏讲故事,这些物品将随孩子一起成长。 **费用:**不等 @@ -130,7 +131,7 @@ **年龄**:5-8 岁 -**这是什么:** 《[Hello Ruby][11]:编码历险记》是 Linda Liukas 的插图书,通过有趣的故事讲述了一个遇到各种问题和朋友(每个都用一个码代表)的女孩,向孩子们介绍了编程概念。Liukas 还有其他副标题为《互联网探险》和《计算机内的旅程》的《Hello Ruby》书籍,而《编码历险记》已以 20 多种语言出版。 +**这是什么:** 《[Hello Ruby][11]:编码历险记》是 Linda Liukas 的插图书,通过有趣的故事讲述了一个遇到各种问题和朋友(每个都用一个码代表)的女孩,向孩子们介绍了编程概念。Liukas 还有其他副标题为《互联网探险》和《计算机内的旅程》的《Hello Ruby》系列书籍,而《编码历险记》已以 20 多种语言出版。 **为什么我喜欢它:**作者在书中附带了许多免费、有趣和无障碍的活动,可以从 Hello Ruby 网站下载和打印这些活动。这些活动教授编码概念、还涉及艺术表达、沟通、甚至时间安排。 @@ -144,7 +145,7 @@ **内容是什么:**由《编程少女》的创始人 Reshma Saujani 撰写,《[编程少女:学会编程和改变世界][13]》为年轻女孩(以及男孩)提供了科技领域的实用信息。它涵盖了广泛的主题,包括编程语言、用例、术语和词汇、职业选择以及技术行业人士的个人简介和访谈。 -**为什么我喜欢它:**本书以讲述了大多数面向成年人的网站没有的技术故事。技术涉及许多学科,对于年轻人来说,重要的是要了解他们可以使用它来解决现实世界中的问题并有所作为。 +**为什么我喜欢它:**本书以讲述了大多数面向成年人的网站都没有的技术故事。这些技术涉及许多学科,对于年轻人来说,重要的是要了解他们可以使用它来解决现实世界中的问题并有所作为。 **成本:**精装书的标价为 17.99 美元,平装书的标价为 10.99 美元,但你可以通过本地或在线书店以更低的价格找到。 @@ -155,7 +156,7 @@ via: https://opensource.com/article/19/11/maker-gifts-kids 作者:[Jess Weichler][a] 选题:[lujun9972][b] 译者:[wxy](https://github.com/wxy) -校对:[校对者ID](https://github.com/校对者ID) +校对:[wxy](https://github.com/wxy) 本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出 From 5fea83c3ca13a218e3c68bf908fdf9c11c93033f Mon Sep 17 00:00:00 2001 From: Xingyu Wang Date: Wed, 8 Jan 2020 14:08:32 +0800 Subject: [PATCH 18/20] PUB @wxy https://linux.cn/article-11762-1.html --- .../20191130 7 maker gifts for kids and teens.md | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) rename {translated/tech => published}/20191130 7 maker gifts for kids and teens.md (98%) diff --git a/translated/tech/20191130 7 maker gifts for kids and teens.md b/published/20191130 7 maker gifts for kids and teens.md similarity index 98% rename from translated/tech/20191130 7 maker gifts for kids and teens.md rename to published/20191130 7 maker gifts for kids and teens.md index 002e97e442..c92bcc29c9 100644 --- a/translated/tech/20191130 7 maker gifts for kids and teens.md +++ b/published/20191130 7 maker gifts for kids and teens.md @@ -1,8 +1,8 @@ [#]: collector: (lujun9972) [#]: translator: (wxy) [#]: reviewer: (wxy) -[#]: publisher: ( ) -[#]: url: ( ) +[#]: publisher: (wxy) +[#]: url: (https://linux.cn/article-11762-1.html) [#]: subject: (7 maker gifts for kids and teens) [#]: via: (https://opensource.com/article/19/11/maker-gifts-kids) [#]: author: (Jess Weichler https://opensource.com/users/cyanide-cupcake) @@ -12,7 +12,7 @@ > 这份礼物指南使你轻松完成节日礼物的采购,它们可给婴儿、儿童、青少年及年龄更大的人们带来创造和创新能力。 -![Gift box opens with colors coming out][1] +![](https://img.linux.net.cn/data/attachment/album/202001/08/140516t4ewey9ryu24tpz5.jpg) 还在纠结这个假期给年轻人买什么礼物?这是我精选的开源礼物,这些礼物将激发未来的创意和灵感。 From 50e8ec4229d440d6ca3ab56aa765b1d1eaaecef7 Mon Sep 17 00:00:00 2001 From: DarkSun Date: Thu, 9 Jan 2020 00:53:24 +0800 Subject: [PATCH 19/20] =?UTF-8?q?=E9=80=89=E9=A2=98:=2020200108=20How=20to?= =?UTF-8?q?=20setup=20multiple=20monitors=20in=20sway?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit sources/tech/20200108 How to setup multiple monitors in sway.md --- ... How to setup multiple monitors in sway.md | 85 +++++++++++++++++++ 1 file changed, 85 insertions(+) create mode 100644 sources/tech/20200108 How to setup multiple monitors in sway.md diff --git a/sources/tech/20200108 How to setup multiple monitors in sway.md b/sources/tech/20200108 How to setup multiple monitors in sway.md new file mode 100644 index 0000000000..f156d204dc --- /dev/null +++ b/sources/tech/20200108 How to setup multiple monitors in sway.md @@ -0,0 +1,85 @@ +[#]: collector: (lujun9972) +[#]: translator: ( ) +[#]: reviewer: ( ) +[#]: publisher: ( ) +[#]: url: ( ) +[#]: subject: (How to setup multiple monitors in sway) +[#]: via: (https://fedoramagazine.org/how-to-setup-multiple-monitors-in-sway/) +[#]: author: (arte219 https://fedoramagazine.org/author/arte219/) + +How to setup multiple monitors in sway +====== + +![][1] + +Sway is a tiling Wayland compositor which has mostly the same features, look and workflow as the [i3 X11 window manager][2]. Because Sway uses Wayland instead of X11, the tools to setup X11 don’t always work in sway. This includes tools like _xrandr_, which are used in X11 window managers or desktops to setup monitors. This is why monitors have to be setup by editing the sway config file, and that’s what this article is about. + +## **Getting your monitor ID’s** + +First, you have to get the names sway uses to refer to your monitors. You can do this by running: + +``` +$ swaymsg -t get_outputs +``` + +You will get information about all of your monitors, every monitor separated by an empty line. + +You have to look for the first line of every section, and for what’s after “Output”. For example, when you see a line like “_Output DVI-D-1 ‘Philips Consumer Electronics Company’_”, the output ID is “DVI-D-1”. Note these ID’s and which physical monitors they belong to. + +## **Editing the config file** + +If you haven’t edited the Sway config file before, you have to copy it to your home directory by running this command: + +``` +cp -r /etc/sway/config ~/.config/sway/config +``` + +Now the default config file is located in _~/.config/sway_ and called “config”. You can edit it using any text editor. + +Now you have to do a little bit of math. Imagine a grid with the origin in the top left corner. The units of the X and Y coordinates are pixels. The Y axis is inverted. This means that if you, for example, start at the origin and you move 100 pixels to the right and 80 pixels down, your coordinates will be (100, 80). + +You have to calculate where your displays are going to end up on this grid. The locations of the displays are specified with the top left pixel. For example, if we want to have a monitor with name HDMI1 and a resolution of 1920×1080, and to the right of it a laptop monitor with name eDP1 and a resolution of 1600×900, you have to type this in your config file: + +``` +output HDMI1 pos 0 0 +output eDP1 pos 1920 0 +``` + +You can also specify the resolutions manually by using the _res_ option:  + +``` +output HDMI1 pos 0 0 res 1920x1080 +output eDP1 pos 1920 0 res 1600x900 +``` + +## **Binding workspaces to monitors** + +Using sway with multiple monitors can be a little bit tricky with workspace management. Luckily, you can bind workspaces to a specific monitor, so you can easily switch to that monitor and use your displays more efficiently. This can simply be done by the workspace command in your config file. For example, if you want to bind workspace 1 and 2 to monitor DVI-D-1 and workspace 8 and 9 to monitor HDMI-A-1, you can do that by using: + +``` +workspace 1 output DVI-D-1 +workspace 2 output DVI-D-1 +``` + +``` +workspace 8 output HDMI-A-1 +workspace 9 output HDMI-A-1 +``` + +That’s it! These are the basics of multi monitor setup in sway. A more detailed guide can be found at . + +-------------------------------------------------------------------------------- + +via: https://fedoramagazine.org/how-to-setup-multiple-monitors-in-sway/ + +作者:[arte219][a] +选题:[lujun9972][b] +译者:[译者ID](https://github.com/译者ID) +校对:[校对者ID](https://github.com/校对者ID) + +本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出 + +[a]: https://fedoramagazine.org/author/arte219/ +[b]: https://github.com/lujun9972 +[1]: https://fedoramagazine.org/wp-content/uploads/2020/01/sway-multiple-monitors-816x345.png +[2]: https://fedoramagazine.org/getting-started-i3-window-manager/ From 233e9833d0e95677b396bf973cd430454f508085 Mon Sep 17 00:00:00 2001 From: DarkSun Date: Thu, 9 Jan 2020 00:53:46 +0800 Subject: [PATCH 20/20] add done: 20200108 How to setup multiple monitors in sway.md --- ...duction to the Linux goto shell utility.md | 599 ++++++++++++++++++ 1 file changed, 599 insertions(+) create mode 100644 sources/tech/20200107 Introduction to the Linux goto shell utility.md diff --git a/sources/tech/20200107 Introduction to the Linux goto shell utility.md b/sources/tech/20200107 Introduction to the Linux goto shell utility.md new file mode 100644 index 0000000000..b683f34449 --- /dev/null +++ b/sources/tech/20200107 Introduction to the Linux goto shell utility.md @@ -0,0 +1,599 @@ +[#]: collector: (lujun9972) +[#]: translator: ( ) +[#]: reviewer: ( ) +[#]: publisher: ( ) +[#]: url: ( ) +[#]: subject: (Introduction to the Linux goto shell utility) +[#]: via: (https://opensource.com/article/20/1/directories-autocomplete-linux) +[#]: author: (Lazarus Lazaridis https://opensource.com/users/iridakos) + +Introduction to the Linux goto shell utility +====== +Learn how to use goto to alias and navigate to directories with +autocomplete in Linux. +![Files in a folder][1] + +The goto shell utility allows users to navigate to aliased directories and also supports autocompletion. + +## How it works + +Before you can use goto, you need to register your directory aliases. For example: + + +``` +`goto -r dev /home/iridakos/development` +``` + +then change to that directory, e.g.: + + +``` +`goto dev` +``` + +![goto demo][2] + +## Autocompletion in goto + +**goto** comes with a nice autocompletion script—whenever you press the Tab key after the **goto** command, Bash or Zsh will prompt you with suggestions of the aliases that are available: + + +``` +$ goto <tab> +bc /etc/bash_completion.d                     +dev /home/iridakos/development +rubies /home/iridakos/.rvm/rubies +``` + +## Installing goto + +There are several ways to install goto. + +### Via script + +Clone the repository and run the install script as a superuser or root: + + +``` +git clone +cd goto +sudo ./install +``` + +### Manually + +Copy the file **goto.sh** somewhere in your filesystem and add a line in your **.zshrc** or **.bashrc** to source it. + +For example, if you placed the file in your home folder, all you have to do is add the following line to your **.zshrc** or **.bashrc** file: + + +``` +`source ~/goto.sh` +``` + +### MacOS Homebrew + +A formula named **goto** is available for the Bash shell in MacOS: + + +``` +`brew install goto` +``` + +### Add colored output + + +``` +`echo -e "\$include /etc/inputrc\nset colored-completion-prefix on" >> ~/.inputrc` +``` + +**Notes:** + + * You need to restart your shell after installation. + * You need to have the Bash completion feature enabled for Bash in MacOS (see this [issue][3]). + * You can install it with **brew install bash-completion** if you don't have it enabled. + + + +## Ways to use goto + +### Change to an aliased directory + +To change to an aliased directory, type: + + +``` +`goto ` +``` + +For example: + + +``` +`goto dev` +``` + +### Register an alias + +To register a directory alias, type: + + +``` +`goto -r ` +``` + +or + + +``` +`goto --register ` +``` + +For example: + + +``` +`goto -r blog /mnt/external/projects/html/blog` +``` + +or + + +``` +`goto --register blog /mnt/external/projects/html/blog` +``` + +**Notes:** + + * **goto** **expands** the directories, so you can easily alias your current directory with the following command and it will automatically be aliased to the whole path: [code]`goto -r last_release .` +``` + * Pressing the Tab key after the alias name provides the shell's default directory suggestions. + + + +### Unregister an alias + +To unregister an alias, use: +``` +`goto -u ` +``` +or +``` +`goto --unregister ` +``` +For example: +``` +`goto -u last_release` +``` +or +``` +`goto --unregister last_release` +``` +**Note:** By pressing the Tab key after the command (**-u** or **\--unregister**), the completion script will prompt you with the list of registered aliases. + +### List aliases + +To get a list of your currently registered aliases, use: +``` +`goto -l` +``` +or +``` +`goto --list` +``` +### Expand an alias + +To expand an alias to its value, use: +``` +`goto -x ` +``` +or +``` +`goto --expand ` +``` +For example: +``` +`goto -x last_release` +``` +or +``` +`goto --expand last_release` +``` +### Clean up aliases + +To clean up the aliases from directories that are no longer accessible in your filesystem, use: +``` +`goto -c` +``` +or +``` +`goto --cleanup` +``` +### Get help + +To view the tool's help information, use: +``` +`goto -h` +``` +or +``` +`goto --help` +``` +### Check the version + +To view the tool's version, use: +``` +`goto -v` +``` +or +``` +`goto --version` +``` +### Push before changing directories + +To push the current directory onto the directory stack before changing directories, type: +``` +`goto -p ` +``` +or +``` +`goto --push ` +``` +### Revert to a pushed directory + +To return to a pushed directory, type: +``` +`goto -o` +``` +or +``` +`goto --pop` +``` +**Note:** This command is equivalent to **popd** but within the **goto** command. + +## Troubleshooting + +If you see the error **command not found: compdef** in Zsh, it means you need to load **bashcompinit**. To do so, append this to your **.zshrc** file: +``` + + +autoload bashcompinit +bashcompinit + +``` +## Get involved + +The goto tool is open source under the [MIT License][4] terms, and contributions are welcomed. To learn more, visit the [Contributing][5] section in goto's GitHub repository. + +## The goto script +``` + + +goto() +{ +  local target +  _goto_resolve_db + +  if [ -z "$1" ]; then +    # display usage and exit when no args +    _goto_usage +    return +  fi + +  subcommand="$1" +  shift +  case "$subcommand" in +    -c|--cleanup) +      _goto_cleanup "$@" +      ;; +    -r|--register) # Register an alias +      _goto_register_alias "$@" +      ;; +    -u|--unregister) # Unregister an alias +      _goto_unregister_alias "$@" +      ;; +    -p|--push) # Push the current directory onto the pushd stack, then goto +      _goto_directory_push "$@" +      ;; +    -o|--pop) # Pop the top directory off of the pushd stack, then change that directory +      _goto_directory_pop +      ;; +    -l|--list) +      _goto_list_aliases +      ;; +    -x|--expand) # Expand an alias +      _goto_expand_alias "$@" +      ;; +    -h|--help) +      _goto_usage +      ;; +    -v|--version) +      _goto_version +      ;; +    *) +      _goto_directory "$subcommand" +      ;; +  esac +  return $? +} + +_goto_resolve_db() +{ +  GOTO_DB="${GOTO_DB:-$HOME/.goto}" +  touch -a "$GOTO_DB" +} + +_goto_usage() +{ +  cat <<\USAGE +usage: goto [<option>] <alias> [<directory>] + +default usage: +  goto <alias> \- changes to the directory registered for the given alias + +OPTIONS: +  -r, --register: registers an alias +    goto -r|--register <alias> <directory> +  -u, --unregister: unregisters an alias +    goto -u|--unregister <alias> +  -p, --push: pushes the current directory onto the stack, then performs goto +    goto -p|--push <alias> +  -o, --pop: pops the top directory from the stack, then changes to that directory +    goto -o|--pop +  -l, --list: lists aliases +    goto -l|--list +  -x, --expand: expands an alias +    goto -x|--expand <alias> +  -c, --cleanup: cleans up non existent directory aliases +    goto -c|--cleanup +  -h, --help: prints this help +    goto -h|--help +  -v, --version: displays the version of the goto script +    goto -v|--version +USAGE +} + +# Displays version +_goto_version() +{ +  echo "goto version 1.2.4.1" +} + +# Expands directory. +# Helpful for ~, ., .. paths +_goto_expand_directory() +{ +  builtin cd "$1" 2>/dev/null && pwd +} + +# Lists registered aliases. +_goto_list_aliases() +{ +  local IFS=$' ' +  if [ -f "$GOTO_DB" ]; then +    while read -r name directory; do +      printf '\e[1;36m%20s  \e[0m%s\n' "$name" "$directory" +    done < "$GOTO_DB" +  else +    echo "You haven't configured any directory aliases yet." +  fi +} + +# Expands a registered alias. +_goto_expand_alias() +{ +  if [ "$#" -ne "1" ]; then +    _goto_error "usage: goto -x|--expand <alias>" +    return +  fi + +  local resolved + +  resolved=$(_goto_find_alias_directory "$1") +  if [ -z "$resolved" ]; then +    _goto_error "alias '$1' does not exist" +    return +  fi + +  echo "$resolved" +} + +# Lists duplicate directory aliases +_goto_find_duplicate() +{ +  local duplicates= + +  duplicates=$(sed -n 's:[^ ]* '"$1"'$:&:p' "$GOTO_DB" 2>/dev/null) +  echo "$duplicates" +} + +# Registers and alias. +_goto_register_alias() +{ +  if [ "$#" -ne "2" ]; then +    _goto_error "usage: goto -r|--register <alias> <directory>" +    return 1 +  fi + +  if ! [[ $1 =~ ^[[:alnum:]]+[a-zA-Z0-9_-]*$ ]]; then +    _goto_error "invalid alias - can start with letters or digits followed by letters, digits, hyphens or underscores" +    return 1 +  fi + +  local resolved +  resolved=$(_goto_find_alias_directory "$1") + +  if [ -n "$resolved" ]; then +    _goto_error "alias '$1' exists" +    return 1 +  fi + +  local directory +  directory=$(_goto_expand_directory "$2") +  if [ -z "$directory" ]; then +    _goto_error "failed to register '$1' to '$2' - can't cd to directory" +    return 1 +  fi + +  local duplicate +  duplicate=$(_goto_find_duplicate "$directory") +  if [ -n "$duplicate" ]; then +    _goto_warning "duplicate alias(es) found: \\\n$duplicate" +  fi + +  # Append entry to file. +  echo "$1 $directory" >> "$GOTO_DB" +  echo "Alias '$1' registered successfully." +} + +# Unregisters the given alias. +_goto_unregister_alias() +{ +  if [ "$#" -ne "1" ]; then +    _goto_error "usage: goto -u|--unregister <alias>" +    return 1 +  fi + +  local resolved +  resolved=$(_goto_find_alias_directory "$1") +  if [ -z "$resolved" ]; then +    _goto_error "alias '$1' does not exist" +    return 1 +  fi + +  # shellcheck disable=SC2034 +  local readonly GOTO_DB_TMP="$HOME/.goto_" +  # Delete entry from file. +  sed "/^$1 /d" "$GOTO_DB" > "$GOTO_DB_TMP" && mv "$GOTO_DB_TMP" "$GOTO_DB" +  echo "Alias '$1' unregistered successfully." +} + +# Pushes the current directory onto the stack, then goto +_goto_directory_push() +{ +  if [ "$#" -ne "1" ]; then +    _goto_error "usage: goto -p|--push <alias>" +    return +  fi + +  { pushd . || return; } 1>/dev/null 2>&1 + +  _goto_directory "$@" +} + +# Pops the top directory from the stack, then goto +_goto_directory_pop() +{ +  { popd || return; } 1>/dev/null 2>&1 +} + +# Unregisters aliases whose directories no longer exist. +_goto_cleanup() +{ +  if ! [ -f "$GOTO_DB" ]; then +    return +  fi + +  while IFS= read -r i && [ -n "$i" ]; do +    echo "Cleaning up: $i" +    _goto_unregister_alias "$i" +  done <<< "$(awk '{al=$1; $1=""; dir=substr($0,2); +                    system("[ ! -d \"" dir "\" ] && echo " al)}' "$GOTO_DB")" +} + +# Changes to the given alias' directory +_goto_directory() +{ +  local target + +  target=$(_goto_resolve_alias "$1") || return 1 + +  builtin cd "$target" 2> /dev/null || \ +    { _goto_error "Failed to goto '$target'" && return 1; } +} + +# Fetches the alias directory. +_goto_find_alias_directory() +{ +  local resolved + +  resolved=$(sed -n "s/^$1 \\\\(.*\\\\)/\\\1/p" "$GOTO_DB" 2>/dev/null) +  echo "$resolved" +} + +# Displays the given error. +# Used for common error output. +_goto_error() +{ +  (>&2 echo -e "goto error: $1") +} + +# Displays the given warning. +# Used for common warning output. +_goto_warning() +{ +  (>&2 echo -e "goto warning: $1") +} + +# Displays entries with aliases starting as the given one. +_goto_print_similar() +{ +  local similar + +  similar=$(sed -n "/^$1[^ ]* .*/p" "$GOTO_DB" 2>/dev/null) +  if [ -n "$similar" ]; then +    (>&2 echo "Did you mean:") +    (>&2 column -t <<< "$similar") +  fi +} + +# Fetches alias directory, errors if it doesn't exist. +_goto_resolve_alias() +{ +  local resolved + +  resolved=$(_goto_find_alias_directory "$1") + +  if [ -z "$resolved" ]; then +    _goto_error "unregistered alias $1" +    _goto_print_similar "$1" +    return 1 +  else +    echo "${resolved}" +  fi +} + +# Completes the goto function with the available commands +_complete_goto_commands() +{ +  local IFS=$' \t\n' + +  # shellcheck disable=SC2207 +  COMPREPLY=($(compgen -W "-r --register -u --unregister -p --push -o --pop -l --list -x --expand -c --cleanup -v --version" -- "$1")) +} + +# Completes the goto function with the available aliases +_complete_goto_aliases() +{ +  local IFS=$'\n' matches +  _goto_resolve_db + +  # shellcheck disable=SC2207 +  matches=($(sed -n "/^$1/p" "$GOTO_DB" 2>/dev/null)) + +  if [ "${#matches[@]}" -eq "1" ]; then +    # remove the filenames attribute from the completion method +    compopt +o filenames 2>/dev/null + +    # if you find only one alias don't append the directory +    COMPREPLY=("${matches[0]// *}") +  else +    for i in "${!matches[@]}"; do +      # remove the filenames attribute from the completion method +      compopt +o filenames 2>/dev/null + +      if ! [[ $(uname -s) =~ Darwin* ]]; then +        matches[$i]=$(printf '%*s' "-$COLUMNS" "${matches[$i]}") + +        COMPREPLY+=("$(compgen -W "${matches[$i]}")") +      els \ No newline at end of file