-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy path2023-05-16T06_50_55.342Z - Master Your Z Shell with These Outrageously Useful Tips - Blog - Reason I Am Here - Nacho Caballero.html
720 lines (463 loc) · 86.5 KB
/
2023-05-16T06_50_55.342Z - Master Your Z Shell with These Outrageously Useful Tips - Blog - Reason I Am Here - Nacho Caballero.html
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
<!DOCTYPE html> <html style lang=en><!--
Page saved with SingleFile
url: http://reasoniamhere.com/2014/01/11/outrageously-useful-tips-to-master-your-z-shell/
saved date: Tue May 16 2023 12:20:55 GMT+0530 (India Standard Time)
--><meta charset=utf-8>
<title>Master Your Z Shell with These Outrageously Useful Tips - Blog - Reason I Am Here - Nacho Caballero</title>
<meta name=author content="Nacho Caballero">
<meta name=viewport content="width=device-width, initial-scale=1.0">
<!--[if lt IE 9]>
<script src="http://html5shim.googlecode.com/svn/trunk/html5.js"></script>
<![endif]-->
<style>/*!
* Bootstrap v2.2.2
*
* Copyright 2012 Twitter, Inc
* Licensed under the Apache License v2.0
* http://www.apache.org/licenses/LICENSE-2.0
*
* Designed and built with all the love in the world @twitter by @mdo and @fat.
*/footer{display:block}html{font-size:100%;-webkit-text-size-adjust:100%;-ms-text-size-adjust:100%}a:focus{outline:thin dotted #333;outline:5px auto -webkit-focus-ring-color;outline-offset:-2px}a:hover,a:active{outline:0}body{margin:0;font-family:"Helvetica Neue",Helvetica,Arial,sans-serif;font-size:14px;line-height:20px;color:#333333;background-color:#ffffff}a{color:#0088cc;text-decoration:none}a:hover{color:#005580;text-decoration:underline}.row-fluid{width:100%}.row-fluid:before,.row-fluid:after{display:table;content:"";line-height:0}.row-fluid:after{clear:both}.row-fluid [class*="span"]{display:block;min-height:30px;-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box;float:left;margin-left:2.127659574468085%}.row-fluid [class*="span"]:first-child{margin-left:0}.row-fluid .span12{width:100%}p{margin:0 0 10px}strong{font-weight:bold}em{font-style:italic}h1,h2,h3{margin:10px 0;font-family:inherit;font-weight:bold;color:inherit;text-rendering:optimizelegibility}h2,h3{line-height:40px}h2{font-size:31.5px}h3{font-size:24.5px}.page-header{padding-bottom:9px;margin:20px 0 30px}ul,ol{padding:0;margin:0 0 10px 25px}li{line-height:20px}ul.inline{margin-left:0;list-style:none}ul.inline>li{display:inline-block;padding-left:5px;padding-right:5px}hr{border-bottom:1px solid #ffffff}code,pre{font-family:Monaco,Menlo,Consolas,"Courier New",monospace;color:#333333;-webkit-border-radius:3px;-moz-border-radius:3px;border-radius:3px}code{color:#d14;background-color:#f7f7f9;border:1px solid #e1e1e8}pre{display:block;word-break:break-all;-webkit-border-radius:4px;-moz-border-radius:4px;border-radius:4px}pre code{background-color:transparent;border:0}[class^="icon-"]{display:inline-block;width:14px;height:14px;line-height:14px;vertical-align:text-top;background-image:/* original URL: http://reasoniamhere.com/assets/themes/twitter/bootstrap/img/glyphicons-halflings.png */url();background-repeat:no-repeat;margin-top:1px}.icon-tags{background-position:-25px -48px}.nav{margin-left:0;margin-bottom:20px;list-style:none}.nav>li>a{display:block}.nav>li>a:hover{text-decoration:none;background-color:#eeeeee}.navbar{overflow:visible;margin-bottom:20px}.navbar-inner{min-height:40px;padding-left:20px;padding-right:20px;background-color:#fafafa;background-image:-moz-linear-gradient(top,#ffffff,#f2f2f2);background-image:-webkit-gradient(linear,0 0,0 100%,from(#ffffff),to(#f2f2f2));background-image:-webkit-linear-gradient(top,#ffffff,#f2f2f2);background-image:-o-linear-gradient(top,#ffffff,#f2f2f2);background-image:linear-gradient(to bottom,#ffffff,#f2f2f2);background-repeat:repeat-x;border:1px solid #d4d4d4;-webkit-border-radius:4px;-moz-border-radius:4px;border-radius:4px;-webkit-box-shadow:0 1px 4px rgba(0,0,0,0.065);-moz-box-shadow:0 1px 4px rgba(0,0,0,0.065);box-shadow:0 1px 4px rgba(0,0,0,0.065)}.navbar-inner:before,.navbar-inner:after{display:table;content:"";line-height:0}.navbar-inner:after{clear:both}.navbar .brand{float:left;display:block;padding:10px 20px 10px;margin-left:-20px;font-size:20px;color:#777777;text-shadow:0 1px 0#ffffff}.navbar .brand:hover{text-decoration:none}.navbar .nav{position:relative;left:0;display:block;margin:0 10px 0 0}.navbar .nav>li{float:left}.navbar .nav>li>a{float:none;padding:10px 15px 10px;color:#777777;text-decoration:none;text-shadow:0 1px 0#ffffff}.navbar .nav>li>a:focus,.navbar .nav>li>a:hover{background-color:transparent;color:#333333;text-decoration:none}@-webkit-keyframes progress-bar-stripes{from{background-position:40px 0}to{background-position:0 0}}@-moz-keyframes progress-bar-stripes{from{background-position:40px 0}to{background-position:0 0}}@-ms-keyframes progress-bar-stripes{from{background-position:40px 0}to{background-position:0 0}}@-o-keyframes progress-bar-stripes{from{background-position:0 0}to{background-position:40px 0}}@keyframes progress-bar-stripes{from{background-position:40px 0}to{background-position:0 0}}@media (max-width:767px){body{padding-left:20px;padding-right:20px}.row-fluid{width:100%}.row-fluid [class*="span"]{float:none;display:block;margin-left:0;-webkit-box-sizing:border-box;-moz-box-sizing:border-box}.row-fluid .span12{width:100%;-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box}}@media (min-width:768px) and (max-width:979px){.row-fluid{width:100%}.row-fluid:before,.row-fluid:after{display:table;content:"";line-height:0}.row-fluid:after{clear:both}.row-fluid [class*="span"]{display:block;min-height:30px;-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box;float:left;margin-left:2.7624309392265194%}.row-fluid [class*="span"]:first-child{margin-left:0}.row-fluid .span12{width:100%}}@media (min-width:1200px){.row-fluid{width:100%}.row-fluid:before,.row-fluid:after{display:table;content:"";line-height:0}.row-fluid:after{clear:both}.row-fluid [class*="span"]{display:block;min-height:30px;-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box;float:left;margin-left:2.564102564102564%}.row-fluid [class*="span"]:first-child{margin-left:0}.row-fluid .span12{width:100%}}@media (max-width:979px){body{padding-top:0}.navbar .brand{padding-left:10px;padding-right:10px;margin:0 0 0-5px}}.container-narrow{margin:0 auto;max-width:900px}.navbar .nav{float:right}.date{font-family:"ff-tisa-web-pro",Georgia,Cambria,"Times New Roman",Times,serif;margin-bottom:20px;font-weight:bold;font-size:22px;line-height:1.45}.tag_box{margin:0;overflow:hidden}.tag_box li i{opacity:0.9}.tag_box.inline li{float:left}.tag_box a{padding:3px 6px;margin:2px;background:#eee;color:#555;border-radius:3px;text-decoration:none;border:1px dashed #cccccc}.tag_box a span{vertical-align:super;font-size:0.8em}.tag_box a:hover{background-color:#e5e5e5}pre .comment{color:#999999}pre .variable{color:#f2777a}pre .string{color:#99cc99}pre .keyword{color:#cc99cc}pre code{display:block;background:#2d2d2d;color:#cccccc}h1{margin-bottom:0px;margin-top:0px;font-size:52px;line-height:1;letter-spacing:-1px;font-family:"freight-sans-pro","Myriad Pro","Lucida Grande","Lucida Sans Unicode","Lucida Sans",Geneva,Verdana,sans-serif;font-weight:700;border-bottom:0}a:hover{text-decoration:none}.page-header{border-bottom:0}.content{padding-top:20px}.content p,.content ul{margin-bottom:30px;font-family:Georgia,Cambria,"Times New Roman",Times,serif;font-weight:400;font-size:22px;line-height:1.45}.content li,.content ol{font-family:Georgia,Cambria,"Times New Roman",Times,serif;font-weight:400;font-size:22px;line-height:1.45}code,pre{padding:0;margin:0 0 0;font-size:18px;line-height:25px;overflow:auto;word-wrap:normal;white-space:pre}p code,ul code{white-space:pre-line;padding:0 5px}pre{margin:30px 0;background-color:transparent;border:0}pre code{white-space:pre;padding:1em}hr{margin:20px 0;border:0;border-top:3px solid #eeeeee}.navbar .brand{font-weight:700}.nav{font-size:18px}.copyright{float:right}footer{padding-top:50px}</style>
<link href=http://reasoniamhere.com/atom.xml type=application/atom+xml rel=alternate title="Sitewide ATOM Feed">
<link href=http://reasoniamhere.com/rss.xml type=application/rss+xml rel=alternate title="Sitewide RSS Feed">
<link type=image/x-icon rel="shortcut icon" href="" data-sf-original-href=http://reasoniamhere.com/favicon.ico><style>.sf-hidden{display:none!important}</style><link rel=canonical href=http://reasoniamhere.com/2014/01/11/outrageously-useful-tips-to-master-your-z-shell/><meta http-equiv=content-security-policy content="default-src 'none'; font-src 'self' data:; img-src 'self' data:; style-src 'unsafe-inline'; media-src 'self' data:; script-src 'unsafe-inline' data:; object-src 'self' data:; frame-src 'self' data:;"><style>img[src="data:,"],source[src="data:,"]{display:none!important}</style></head>
<body>
<div class=navbar>
<div class=navbar-inner>
<div class=container-narrow>
<a class=brand href=http://reasoniamhere.com/>Reason I am here</a>
<ul class=nav>
<li><a href=http://reasoniamhere.com/about.html>Nacho Caballero</a></li>
<li><a href=http://reasoniamhere.com/archive.html>Article Index</a></li>
<li><a href=http://reasoniamhere.com/tags.html>Tags</a></li>
</ul>
</div>
</div>
</div>
<div class=container-narrow>
<div class=content>
<div class=page-header>
<h1>Master Your Z Shell with These Outrageously Useful Tips </h1>
</div>
<div class="row-fluid post-full">
<div class=span12>
<span class=date>11 January 2014</span>
<div class=content>
<p>If you had previously installed Zsh but never got around to exploring all of its magic features, this post is for you.</p>
<p>If you never thought of using a different shell than the one that came by default when you got your computer, I recommend you go out and check the Z shell. Here are some <a href=http://blog.coolaj86.com/articles/zsh-is-to-bash-as-vim-is-to-vi.html>Linux</a> <a href=http://linuxg.net/how-to-install-zsh-shell-how-to-set-it-as-a-default-login-shell/>guides</a> that explain how to install it and set it as your default shell. You probably have Zsh installed you are on a Mac, but there’s nothing like the warm fuzzy feeling of running the latest version (here’s <a href=http://zanshin.net/2013/09/03/how-to-use-homebrew-zsh-instead-of-max-os-x-default/>a way to upgrade</a> using Homebrew).</p>
<p>While you’re at it, you should also get <strong>oh-my-zsh</strong>, a framework that makes Zsh easier to configure. It’s pretty easy to install, just run this:</p>
<div class=highlight><pre><code class=language-bash data-lang=bash>curl -L https://raw.github.com/robbyrussell/oh-my-zsh/master/tools/install.sh <span class=p>|</span> sh
<span class=nb><span class=keyword>echo</span></span> <span class=nv><span class=variable>$0</span></span>
<span class=c><span class=comment># if you don't see /bin/zsh you might need</span></span>
<span class=c><span class=comment># to open up a new window, or manually run: zsh</span></span>
</code></pre></div>
<p><hr><p></p>
<p>The Zsh manual is a daunting beast. Just the <a href=http://zsh.sourceforge.net/Doc/Release/Expansion.html>chapter on expansions</a> has 32 subsections. Forget about memorizing this madness in one sitting. Instead, we’ll focus on understanding a few useful concepts, and referencing the manual for additional help.</p>
<p>The three main sections of this post are <strong>file picking</strong>, <strong>variable transformations</strong>, and <strong>magic tabbing</strong>. If you’re pressed for time, read the beginning of each one, and come back later to soak up the details (make sure you stick around for the bonus tips at the end).</p>
<h2>You only learn by doing</h2>
<p>Reading this post will only take you 10% of the way into Zsh paradise; to really grok what it’s all about, you need to run the commands yourself. I’m giving you everything you need to create the file structure that we’ll be using for the entire post. Simply copy and paste this into your Zsh window:</p>
<div class=highlight><pre><code class=language-bash data-lang=bash><span class=c><span class=comment># run me to get the party started</span></span>
<span class=c><span class=comment># create the folder structure</span></span>
mkdir -p zsh_demo/<span class=o>{</span>data,calculations<span class=o>}</span>/africa/<span class=o>{</span>kenya,malawi<span class=o>}</span>/ zsh_demo/<span class=o>{</span>data,calculations<span class=o>}</span>/europe/<span class=o>{</span>malta,poland<span class=o>}</span>/ zsh_demo/<span class=o>{</span>data,calculations<span class=o>}</span>/asia/<span class=o>{</span>nepal,laos<span class=o>}</span>/
<span class=c><span class=comment># create dummy files inside the data folder</span></span>
<span class=k><span class=keyword>for</span></span> country_folder <span class=keyword>in</span> zsh_demo/data/*/*<span class=p>;</span> <span class=k><span class=keyword>do</span></span>
dd <span class=k><span class=keyword>if</span></span><span class=o>=</span>/dev/zero <span class=nv>of</span><span class=o>=</span><span class=s2><span class=string>"<span class=variable>${country_folder}</span>/population.txt"</span></span> <span class=nv>bs</span><span class=o>=</span><span class=m>1024</span> <span class=nv>count</span><span class=o>=</span>1
dd <span class=k><span class=keyword>if</span></span><span class=o>=</span>/dev/zero <span class=nv>of</span><span class=o>=</span><span class=s2><span class=string>"<span class=variable>${country_folder}</span>/income.txt"</span></span> <span class=nv>bs</span><span class=o>=</span><span class=m>2048</span> <span class=nv>count</span><span class=o>=</span>1
dd <span class=k><span class=keyword>if</span></span><span class=o>=</span>/dev/zero <span class=nv>of</span><span class=o>=</span><span class=s2><span class=string>"<span class=variable>${country_folder}</span>/literacy.txt"</span></span> <span class=nv>bs</span><span class=o>=</span><span class=m>4096</span> <span class=nv>count</span><span class=o>=</span>1
<span class=c><span class=comment># we say these are dummy files because they don't have any content,</span></span>
<span class=c><span class=comment># but we are making them occupy disk space</span></span>
<span class=k><span class=keyword>done</span></span>
<span class=c><span class=comment># create dummy files inside the calculations folder</span></span>
<span class=k><span class=keyword>for</span></span> country_folder <span class=keyword>in</span> zsh_demo/calculations/*/*<span class=p>;</span> <span class=k><span class=keyword>do</span></span>
touch <span class=s2><span class=string>"<span class=variable>${country_folder}</span>/population_by_province.txt"</span></span> <span class=c><span class=comment># this file is empty</span></span>
dd <span class=k><span class=keyword>if</span></span><span class=o>=</span>/dev/zero <span class=nv>of</span><span class=o>=</span><span class=s2><span class=string>"<span class=variable>${country_folder}</span>/median_income.txt"</span></span> <span class=nv>bs</span><span class=o>=</span><span class=m>2048</span> <span class=nv>count</span><span class=o>=</span>1
dd <span class=k><span class=keyword>if</span></span><span class=o>=</span>/dev/zero <span class=nv>of</span><span class=o>=</span><span class=s2><span class=string>"<span class=variable>${country_folder}</span>/literacy_index.txt"</span></span> <span class=nv>bs</span><span class=o>=</span><span class=m>4096</span> <span class=nv>count</span><span class=o>=</span>1
<span class=k><span class=keyword>done</span></span>
<span class=c><span class=comment># because all the files are nested within the zsh_demo folder you will</span></span>
<span class=c><span class=comment># be able to easily delete them by running:</span></span>
<span class=c><span class=comment># rm -r zsh_demo</span></span>
</code></pre></div>
<p>Your file structure should look like this <code>zsh_demo/{data,calculations}/{continent}{/country}/{file}.txt</code>:</p>
<div class=highlight><pre><code class=language-bash data-lang=bash>zsh_demo
├── data
│ ├── africa
│ │ ├── kenya
│ │ │ ├── literacy.txt
│ │ │ ├── income.txt
│ │ │ └── population.txt
│ │ └── ...
│ ├── asia
│ │ ├── ...
│ └── europe
│ ├── ...
└── data
├── africa
│ ├── kenya
│ │ ├── literacy_index.txt
│ │ ├── median_income.txt
│ │ └── population_by_province.txt
│ └── ...
├── ...
</code></pre></div>
<p>Although you might not initially care about continents and countries, try to relate the examples we’ll be looking at with the type of file structure you usually work with.</p>
<h2>1. File picking</h2>
<p>Warning for purists: some of the features I talk about are not exclusive to Zsh, but I have explain them anyway before we can move on to sexier commands.</p>
<p>First off, <strong>globbing</strong>! A glob is a short expression that lets you select a bunch of files. 99% of the time, there’s an asterisk involved.</p>
<div class=highlight><pre><code class=language-bash data-lang=bash><span class=c><span class=comment># run me please</span></span>
ls zsh_demo/**/*.txt <span class=c><span class=comment># <= this is a glob</span></span>
</code></pre></div>
<p>Globs get replaced by the names of the files that match the glob expression. For example, the glob above lists every text file located anywhere in the <code>zsh_demo</code> folder. Let’s break it down to see what each part is doing:</p>
<div class=highlight><pre><code class=language-bash data-lang=bash><span class=c><span class=comment># list every file directly below the zsh_demo folder</span></span>
ls zsh_demo
<span class=c><span class=comment># list every file in the folders directly below the zsh_demo folder</span></span>
ls zsh_demo/*
<span class=c><span class=comment># list every file in every folder two levels below the zsh_demo folder</span></span>
ls zsh_demo/*/*
<span class=c><span class=comment># list every file anywhere below the zsh_demo folder</span></span>
ls zsh_demo/**/*
<span class=c><span class=comment># list every file that ends in .txt in every folder at any level below the zsh_demo folder</span></span>
ls zsh_demo/**/*.txt
</code></pre></div>
<p>We already used a glob when we specified the location of each file:</p>
<div class=highlight><pre><code class=language-bash data-lang=bash><span class=k><span class=keyword>for</span></span> country_folder <span class=keyword>in</span> zsh_demo/data/*/*<span class=p>;</span> <span class=k><span class=keyword>do</span></span>
<span class=c><span class=comment># create data files for each country</span></span>
<span class=k><span class=keyword>done</span></span>
</code></pre></div>
<p>This loop runs six times. See for yourself:</p>
<div class=highlight><pre><code class=language-bash data-lang=bash>print -l zsh_demo/data/*/*
<span class=c><span class=comment># zsh_demo/africa/kenya</span></span>
<span class=c><span class=comment># zsh_demo/africa/malawi</span></span>
<span class=c><span class=comment># zsh_demo/asia/laos</span></span>
<span class=c><span class=comment># zsh_demo/asia/nepal</span></span>
<span class=c><span class=comment># zsh_demo/europe/malta</span></span>
<span class=c><span class=comment># zsh_demo/europe/poland</span></span>
<span class=c><span class=comment># you could use echo instead of print -l, but the folders would be</span></span>
<span class=c><span class=comment># separated by spaces instead of newlines</span></span>
</code></pre></div>
<h3>Glob operators</h3>
<p>So, what else can you stick inside a glob besides asterisks? Glance at section <a href=http://zsh.sourceforge.net/Doc/Release/Expansion.html#Filename-Generation>14.8.1 of the manual</a> if you want to know all the options. Here are the ones that I find most useful:</p>
<div class=highlight><pre><code class=language-bash data-lang=bash><span class=c><span class=comment># list text files that end in a number from 1 to 10</span></span>
ls -l zsh_demo/**/*<1-10>.txt
<span class=c><span class=comment># list text files that start with the letter a</span></span>
ls -l zsh_demo/**/<span class=o>[</span>a<span class=o>]</span>*.txt
<span class=c><span class=comment># list text files that start with either ab or bc</span></span>
ls -l zsh_demo/**/<span class=o>(</span>ab<span class=p>|</span>bc<span class=o>)</span>*.txt
<span class=c><span class=comment># list text files that don't start with a lower or uppercase c</span></span>
ls -l zsh_demo/**/<span class=o>[</span>^cC<span class=o>]</span>*.txt
</code></pre></div>
<h3>Glob qualifiers</h3>
<p>Now that we got the basic stuff out of the way, let’s dive a little deeper. We previously mentioned this glob: <code>zsh_demo/**/*</code>, which lists every file anywhere below the <code>zsh_demo</code> folder. But, what if we only want to list folders and not regular files, or vice versa? What if we only want to list files bigger than 3 KB? Or maybe, just the last modified file? You can do all that in Zsh using <strong>glob qualifiers</strong>:</p>
<div class=highlight><pre><code class=language-bash data-lang=bash><span class=c><span class=comment># show only directories</span></span>
print -l zsh_demo/**/*<span class=o>(</span>/<span class=o>)</span>
<span class=c><span class=comment># show only regular files</span></span>
print -l zsh_demo/**/*<span class=o>(</span>.<span class=o>)</span>
<span class=c><span class=comment># show empty files</span></span>
ls -l zsh_demo/**/*<span class=o>(</span>L0<span class=o>)</span>
<span class=c><span class=comment># show files greater than 3 KB</span></span>
ls -l zsh_demo/**/*<span class=o>(</span>Lk+3<span class=o>)</span>
<span class=c><span class=comment># show files modified in the last hour</span></span>
print -l zsh_demo/**/*<span class=o>(</span>mh-1<span class=o>)</span>
<span class=c><span class=comment># sort files from most to least recently modified and show the last 3</span></span>
ls -l zsh_demo/**/*<span class=o>(</span>om<span class=o>[</span>1,3<span class=o>])</span>
</code></pre></div>
<p>Glob qualifiers are surrounded in parentheses <code>()</code>, and appear at the end of a glob to make it more stringent. Globs filter files by their name, and glob qualifiers filter by any other attribute (file type, size, modification date). They can be a bit confusing if you don’t know the syntax. Consider this glob qualifier:</p>
<div class=highlight><pre><code class=language-bash data-lang=bash>ls -l zsh_demo/**/*<span class=o>(</span>.Lm-2mh-1om<span class=o>[</span>1,3<span class=o>])</span>
<span class=c><span class=comment># you won't typically write at this level of obfuscation</span></span>
ls -l zsh_demo/**/*<span class=o>(</span>. Lm-2 mh-1 om <span class=o>[</span>1,3<span class=o>])</span>
<span class=c><span class=comment># this is more parseable, but unfortunately Zsh doesn't allow spaces</span></span>
<span class=c><span class=comment># between qualifiers, so you'll get an error</span></span>
</code></pre></div>
<p>You need to be familiar with the individual options to make sense of this madness. Five different things are going on at the same time:</p>
<ol>
<li>The <code>.</code> tells the glob to only show <strong>regular files</strong> (no directories, symbolic links, or other types of files).</li>
<li>The <code>Lm-2</code> tells the glob to show files smaller than 2 MB.
<ul>
<li>Use <code>-</code> for smaller, and <code>+</code> for greater; don’t use anything if you want to specify the exact size (<code>Lm2</code>).</li>
<li>Use <code>m</code> for megabytes, <code>k</code> for kilobytes, or nothing for just bytes (notice that these letters must appear <strong>before</strong> the sign).</li>
</ul></li>
<li>The <code>mh-1</code> tells the glob to show files modified in the last hour
<ul>
<li>Use <code>-</code> if you want files modified within the last X units of time, and <code>+</code> for files modified more than X units of time ago.</li>
<li>Use <code>M</code> for Months, <code>w</code> for weeks, <code>h</code> for hours, <code>m</code> for minutes, and <code>s</code> for seconds (notice that these leters must appear <strong>before</strong> the sign).</li>
</ul></li>
<li>The <code>om</code> tells the glob to sort the remaining files by their modification date.
<ul>
<li>A lowercase <code>o</code> sorts by most recent first, to use the reverse order, make it uppercase <code>O</code>.</li>
<li>Use <code>m</code> to sort by modification date, and <code>L</code> to sort by size (<code>oL</code>).</li>
</ul></li>
<li>The <code>[1,3]</code> tells the glob to show the first 3 files (since we just sorted the files, these will be the most recently modified ones).
<ul>
<li>You can also show a single file (for example, the second one <code>[2]</code>)</li>
</ul></li>
</ol>
<p>Syntax is a trade-off between terseness and obscurity. To people that are familiar with the Zsh jargon, the ability to combine five different filters by only typing a few characters is an awesome feature; to everybody else, it’s incomprehensible mumbo-jumbo. Fortunately, useful shortcuts get used more often, and they become easier to remember.</p>
<p>Head over to <a href=http://zsh.sourceforge.net/Doc/Release/Expansion.html#Filename-Generation>section 14.8.7 of the manual</a> if you’d like to be showered in details.</p>
<p><strong>Pro Tip</strong></p>
<p>Here’s a cool tip for all you advanced devils (feel free to skip to section 2 if you’ve had enough file pickin’ for a day). How can we select folders that don’t contain a given file? In the manual you’ll find information about a qualifier called <code>estring</code>, which runs the code specified by the string, and only keeps the file names that return <em>true</em>. For example:</p>
<div class=highlight><pre><code class=language-bash data-lang=bash><span class=c><span class=comment># show every continent that doesn't contain a country named malta</span></span>
print -l zsh_demo/*/*<span class=o>(</span>e:<span class=s1><span class=string>'[[ ! -e $REPLY/malta ]]'</span></span>:<span class=o>)</span>
<span class=c><span class=comment># zsh_demo/calculations/africa</span></span>
<span class=c><span class=comment># zsh_demo/calculations/asia</span></span>
<span class=c><span class=comment># zsh_demo/data/africa</span></span>
<span class=c><span class=comment># zsh_demo/data/asia</span></span>
</code></pre></div>
<p>Let’s parse this magic:</p>
<ul>
<li>After the <code>e</code>, the <code>string</code> has to be delimited by a convenient character (in this case, a colon <code>:</code>), and the code must be surrounded by single quotes <code>'</code>, so the actual command is just <code>[[ ! -e $REPLY/malta ]]</code>.</li>
<li>The <code>$REPLY</code> variable contains every file name of the ones specified by the glob <code>zsh_demo/*/*</code> in turn, but only a single file at a time.</li>
<li><code>[[ -e file ]]</code> is a <a href=http://zsh.sourceforge.net/Doc/Release/Conditional-Expressions.html#Conditional-Expressions>conditional expression</a> that returns <em>true</em> if the file exists. We want it to return <em>true</em> when the file called <code>malta</code> <em>doesn’t</em> exist, so we reverse it with <code>!</code>.</li>
<li>When the code is executed, the <code>$REPLY</code> variable takes the value of the next file and the code is executed again.</li>
</ul>
<h2>2. Variable transformations</h2>
<h3>Modifiers</h3>
<p>To complicate things even further (or to make them more awesome, depending on your perspective), you can stick one more thing inside the parentheses at the end of your globs: <strong>modifiers</strong>.</p>
<p>Each modifiers is preceded by a colon <code>:</code>, which makes them easily distinguishable from qualifiers.</p>
<div class=highlight><pre><code class=language-bash data-lang=bash><span class=c><span class=comment># A plain old glob</span></span>
print -l zsh_demo/data/europe/poland/*.txt
<span class=c><span class=comment># Return the file name (t stands for tail)</span></span>
print -l zsh_demo/data/europe/poland/*.txt<span class=o>(</span>:t<span class=o>)</span>
<span class=c><span class=comment># Return the file name without the extension (r stands for remove_extension, I think)</span></span>
print -l zsh_demo/data/europe/poland/*.txt<span class=o>(</span>:t:r<span class=o>)</span>
<span class=c><span class=comment># Return the extension</span></span>
print -l zsh_demo/data/europe/poland/*.txt<span class=o>(</span>:e<span class=o>)</span>
<span class=c><span class=comment># Return the parent folder of the file (h stands for head)</span></span>
print -l zsh_demo/data/europe/poland/*.txt<span class=o>(</span>:h<span class=o>)</span>
<span class=c><span class=comment># Return the parent folder of the parent</span></span>
print -l zsh_demo/data/europe/poland/*.txt<span class=o>(</span>:h:h<span class=o>)</span>
<span class=c><span class=comment># Return the parent folder of the first file</span></span>
print -l zsh_demo/data/europe/poland/*.txt<span class=o>([</span>1<span class=o>]</span>:h<span class=o>)</span>
<span class=c><span class=comment># Remember you can combine qualifiers and modifiers.</span></span>
</code></pre></div>
<p>Modifiers are not only for globs, you can also use them with variables (the technical term is <a href=http://zsh.sourceforge.net/Doc/Release/Expansion.html#Parameter-Expansion>parameter expansion</a>):</p>
<div class=highlight><pre><code class=language-bash data-lang=bash><span class=c><span class=comment># run me, you'll like it</span></span>
<span class=nv>my_file</span><span class=o>=(</span>zsh_demo/data/europe/poland/*.txt<span class=o>([</span>1<span class=o>]))</span>
<span class=c><span class=comment># If you want to store a glob in a variable, you must use parentheses</span></span>
print -l <span class=nv><span class=variable>$my_file</span></span>
print -l <span class=nv><span class=variable>$my_file</span></span><span class=o>(</span>:h<span class=o>)</span> <span class=c><span class=comment># this is the syntax we saw before</span></span>
print -l <span class=k><span class=variable>${</span></span><span class=variable><span class=nv>my_file</span><span class=p>:</span><span class=nv>h</span><span class=k>}</span></span><span class=k></span> <span class=c><span class=comment># I find this syntax more convenient</span></span>
print -l <span class=k><span class=variable>${</span></span><span class=variable><span class=nv>my_file</span><span class=p>(:</span><span class=nv>h</span><span class=p>)</span><span class=k>}</span></span><span class=k></span> <span class=c><span class=comment># don't mix the two, or you'll get an error</span></span>
print -l <span class=k><span class=variable>${</span></span><span class=variable><span class=nv>my_file</span><span class=p>:</span><span class=nv>u</span><span class=k>}</span></span><span class=k></span> <span class=c><span class=comment># the :u modifier makes the text uppercase</span></span>
</code></pre></div>
<p>Let’s say we wanted to calculate the maximum income for each country, and store it in a file named <code>{country}_max_income.txt</code> in the corresponding calculations folder. We can do this easily using my favorite modifier (<code>:s</code>):</p>
<div class=highlight><pre><code class=language-bash data-lang=bash><span class=c><span class=comment># run me if you like to run things</span></span>
<span class=k><span class=keyword>for</span></span> file <span class=keyword>in</span> zsh_demo/data/**/income.txt <span class=p>;</span> <span class=k><span class=keyword>do</span></span>
<span class=nv>output_dir</span><span class=o>=</span><span class=k><span class=variable>${</span></span><span class=variable><span class=nv>file</span><span class=p>:</span><span class=nv>h</span><span class=p>:</span><span class=nv>s</span><span class=p>/data/calculations/</span><span class=k>}</span></span><span class=k></span>
<span class=nv>country</span><span class=o>=</span><span class=k><span class=variable>${</span></span><span class=variable><span class=nv>output_dir</span><span class=p>:</span><span class=nv>t</span><span class=k>}</span></span><span class=k></span>
<span class=nv>output_file</span><span class=o>=</span><span class=s2><span class=string>"<span class=variable>${output_dir}</span>/<span class=variable>${country}</span>_max_income.txt"</span></span>
<span class=nb><span class=keyword>echo</span></span> <span class=s2><span class=string>"The max salary is <span class=variable>$RANDOM</span> dollars"</span></span> > <span class=nv><span class=variable>$output_file</span></span>
<span class=k><span class=keyword>done</span></span>
<span class=c><span class=comment># let's see what we just did</span></span>
grep <span class=s2><span class=string>""</span></span> zsh_demo/calculations/**/*_max_income.txt
</code></pre></div>
<p>Note: The <code>grep "" bunch_of_files</code> command is a quick-and-dirty way to show the name of each file and its contents (we could have also used <code>head bunch_of_files</code>, try it).</p>
<p>So, what’s going on here?</p>
<ul>
<li>Each time the <strong>for loop</strong> runs, the <code>$file</code> variable is set to a different income file: <code>zsh_demo/data/africa/kenya/income.txt</code>.</li>
<li>We use the <code>:h</code> modifier to get rid of the file name: <code>zsh_demo/data/africa/kenya/</code>,</li>
<li>and then we use the <code>:s</code> modifier to substitute <code>data</code> with <code>calculations</code>: <code>zsh_demo/calculations/africa/kenya/</code>,</li>
<li>then we store that substituted path in the <code>$output_dir</code> variable.</li>
<li>We use the <code>:t</code> modifier to get the name of the country (<code>kenya</code>) and we store it in the <code>$country</code> variable</li>
<li>Then we stick a slash <code>/</code> between the <code>$output_dir</code> and <code>$country</code> variables, and append <code>_max_income.txt</code> to get our output file path: <code>zsh_demo/calculations/africa/kenya/kenya_max_income.txt</code></li>
<li>The <code>$RANDOM</code> variable gives you a random number every time you call it (it’s just a quick way of generating some content).</li>
<li>The right arrow <code>></code> saves the calculation to the output file.</li>
</ul>
<p><hr><p></p>
<p>A few more things about the <code>:s</code> modifier:</p>
<ul>
<li>You can use any character to separate the <code>:s</code> and the strings:</li>
</ul>
<div class=highlight><pre><code class=language-bash data-lang=bash><span class=c><span class=comment># run me and see for yourself</span></span>
<span class=nv>my_variable</span><span class=o>=</span><span class=s2><span class=string>"path/abcd"</span></span>
<span class=nb><span class=keyword>echo</span></span> <span class=k><span class=variable>${</span></span><span class=variable><span class=nv>my_variable</span><span class=p>:</span><span class=nv>s</span><span class=p>/bc/BC/</span><span class=k>}</span></span><span class=k></span> <span class=c><span class=comment># path/aBCd</span></span>
<span class=nb><span class=keyword>echo</span></span> <span class=k><span class=variable>${</span></span><span class=variable><span class=nv>my_variable</span><span class=p>:</span><span class=nv>s_bc_BC_</span><span class=k>}</span></span><span class=k></span> <span class=c><span class=comment># path/aBCd</span></span>
<span class=nb><span class=keyword>echo</span></span> <span class=k><span class=variable>${</span></span><span class=variable><span class=nv>my_variable</span><span class=p>:</span><span class=nv>s</span><span class=p>/\//./</span><span class=k>}</span></span><span class=k></span> <span class=c><span class=comment># path.abcd (escaping the slash \/)</span></span>
<span class=nb><span class=keyword>echo</span></span> <span class=k><span class=variable>${</span></span><span class=variable><span class=nv>my_variable</span><span class=p>:</span><span class=nv>s_</span><span class=p>/_._</span><span class=k>}</span></span><span class=k></span> <span class=c><span class=comment># path.abcd (slightly more readable)</span></span>
<span class=c><span class=comment># this is useful when you want to substitute the slash (/)</span></span>
<span class=c><span class=comment># without having to escape it with a backslash (\)</span></span>
</code></pre></div>
<ul>
<li>The <code>:s</code> modifier only performs one substitution, if you want to do more, use the <code>:gs</code> modifier (g stands for global)</li>
</ul>
<div class=highlight><pre><code class=language-bash data-lang=bash><span class=c><span class=comment># run me just for fun</span></span>
<span class=nv>my_variable</span><span class=o>=</span><span class=s2><span class=string>"aaa"</span></span>
<span class=nb><span class=keyword>echo</span></span> <span class=k><span class=variable>${</span></span><span class=variable><span class=nv>my_variable</span><span class=p>:</span><span class=nv>s</span><span class=p>/a/A/</span><span class=k>}</span></span><span class=k></span> <span class=c><span class=comment># Aaa</span></span>
<span class=nb><span class=keyword>echo</span></span> <span class=k><span class=variable>${</span></span><span class=variable><span class=nv>my_variable</span><span class=p>:</span><span class=nv>gs</span><span class=p>/a/A/</span><span class=k>}</span></span><span class=k></span> <span class=c><span class=comment># AAA</span></span>
</code></pre></div>
<h3>Expansion flags</h3>
<p>Now that you’ve learned all about glob operators, glob qualifiers, and modifiers, let’s add one more spice to the pot: <strong>expansion flags</strong>.</p>
<div class=highlight><pre><code class=language-bash data-lang=bash><span class=c><span class=comment># run me, it's free</span></span>
<span class=c><span class=comment># Let's say somebody gave you these updated files</span></span>
<span class=c><span class=comment># and told you to replace the old ones</span></span>
<span class=nb><span class=keyword>echo</span></span> <span class=nv><span class=variable>$RANDOM</span></span> > zsh_demo/africa_malawi_population_2014.txt
<span class=nb><span class=keyword>echo</span></span> <span class=nv><span class=variable>$RANDOM</span></span> > zsh_demo/asia_nepal_income_2014.txt
<span class=nb><span class=keyword>echo</span></span> <span class=nv><span class=variable>$RANDOM</span></span> > zsh_demo/europe_malta_literacy_2014.txt
<span class=c><span class=comment># How would you move them to their appropriate folders?</span></span>
<span class=c><span class=comment># Try this wizardry</span></span>
<span class=k><span class=keyword>for</span></span> file <span class=keyword>in</span> zsh_demo/*.txt<span class=p>;</span> <span class=k><span class=keyword>do</span></span>
<span class=nv>file_info</span><span class=o>=(</span><span class=k><span class=variable>${</span></span><span class=variable><span class=p>(s._.)file:</span><span class=nv>t</span><span class=k>}</span></span><span class=k></span><span class=o>)</span>
<span class=nv>continent</span><span class=o>=</span><span class=nv><span class=variable>$file_info</span></span><span class=o>[</span>1<span class=o>]</span>
<span class=nv>country</span><span class=o>=</span><span class=nv><span class=variable>$file_info</span></span><span class=o>[</span>2<span class=o>]</span>
<span class=nv>data</span><span class=o>=</span><span class=nv><span class=variable>$file_info</span></span><span class=o>[</span>3<span class=o>]</span>
mv -f <span class=nv><span class=variable>$file</span></span> zsh_demo/data/<span class=k><span class=variable>${</span></span><span class=variable><span class=nv>continent</span><span class=k>}</span></span><span class=k></span>/<span class=k><span class=variable>${</span></span><span class=variable><span class=nv>country</span><span class=k>}</span></span><span class=k></span>/<span class=k><span class=variable>${</span></span><span class=variable><span class=nv>data</span><span class=k>}</span></span><span class=k></span>.txt
<span class=k><span class=keyword>done</span></span>
<span class=c><span class=comment># Check the contents of the files (.) modified (m) in the last</span></span>
<span class=c><span class=comment># 5 minutes (m-5) to see what you just did</span></span>
grep <span class=s2><span class=string>""</span></span> zsh_demo/**/*<span class=o>(</span>.mm-5<span class=o>)</span>
</code></pre></div>
<p>Let’s tear down the example to understand what’s going on.</p>
<ul>
<li>We are using a <strong>for loop</strong> to cycle through the new text files, and we’re storing each file name in the <code>$file</code> variable.</li>
</ul>
<div class=highlight><pre><code class=language-bash data-lang=bash><span class=nb><span class=keyword>echo</span></span> <span class=nv><span class=variable>$file</span></span>
<span class=c><span class=comment># zsh_demo/europe_malta_literacy_2014.txt</span></span>
</code></pre></div>
<ul>
<li>We don’t want the whole path, so we use the <code>:t</code> modifier to get rid of everything to the left of the first slash <code>/</code>.</li>
</ul>
<div class=highlight><pre><code class=language-bash data-lang=bash><span class=nb><span class=keyword>echo</span></span> <span class=k><span class=variable>${</span></span><span class=variable><span class=nv>file</span><span class=p>:</span><span class=nv>t</span><span class=k>}</span></span><span class=k></span>
<span class=c><span class=comment># europe_malta_literacy_2014.txt</span></span>
</code></pre></div>
<ul>
<li>We use the <code>(s)</code> expansion flag to split the file name at each underscore <code>_</code>.</li>
</ul>
<div class=highlight><pre><code class=language-bash data-lang=bash><span class=nb><span class=keyword>echo</span></span> <span class=k><span class=variable>${</span></span><span class=variable><span class=p>(s._.)file:</span><span class=nv>t</span><span class=k>}</span></span><span class=k></span>
<span class=c><span class=comment># europe malta literacy 2014.txt</span></span>
</code></pre></div>
<ul>
<li>We surround everything with parentheses so we can save it into an array variable.</li>
</ul>
<div class=highlight><pre><code class=language-bash data-lang=bash><span class=nv>file_info</span><span class=o>=</span><span class=k><span class=variable>${</span></span><span class=variable><span class=p>(s._.)file:</span><span class=nv>t</span><span class=k>}</span></span><span class=k></span>
<span class=nb><span class=keyword>echo</span></span> <span class=nv><span class=variable>$file_info</span></span>
<span class=c><span class=comment># the parentheses are missing, file_info contains the wrong information:</span></span>
<span class=c><span class=comment># europe_malta_literacy_2014.txt</span></span>
<span class=nv>file_info</span><span class=o>=(</span><span class=k><span class=variable>${</span></span><span class=variable><span class=p>(s._.)file:</span><span class=nv>t</span><span class=k>}</span></span><span class=k></span><span class=o>)</span>
<span class=nb><span class=keyword>echo</span></span> <span class=nv><span class=variable>$file_info</span></span>
<span class=c><span class=comment># with parentheses, it works:</span></span>
<span class=c><span class=comment># europe malta literacy 2014.txt</span></span>
</code></pre></div>
<ul>
<li>We use an auxiliary variable for continent, country, and data. Since <code>$file_info</code> is now an array, we can refer to its elements by using a numeric index.</li>
</ul>
<div class=highlight><pre><code class=language-bash data-lang=bash><span class=nb><span class=keyword>echo</span></span> <span class=k><span class=variable>${</span></span><span class=variable><span class=nv>file_info</span><span class=p>[3]</span><span class=k>}</span></span><span class=k></span>
<span class=c><span class=comment># literacy</span></span>
</code></pre></div>
<ul>
<li>We use the auxiliary variables to specify the path where we want to move the new files <code>zsh_demo/data/${continent}/${country}/${data}.txt</code>.</li>
</ul>
<p>There are a bunch of other flags described in the <a href=http://zsh.sourceforge.net/Doc/Release/Expansion.html#Parameter-Expansion>14.3.1 section of the manual</a>. Check them out if you’re curious. We have already covered the split expansion flag <code>(s)</code>; the only other one I use is the join expansion flag <code>(j)</code>, which does the opposite of the split flag.</p>
<div class=highlight><pre><code class=language-bash data-lang=bash><span class=nv>my_array</span><span class=o>=(</span>a b c d<span class=o>)</span>
<span class=nb><span class=keyword>echo</span></span> <span class=k><span class=variable>${</span></span><span class=variable><span class=p>(j.-.)my_array</span><span class=k>}</span></span><span class=k></span>
<span class=c><span class=comment># a-b-c-d</span></span>
<span class=c><span class=comment># Since we are joining using dots (.), it makes more sense to</span></span>
<span class=c><span class=comment># use underscores (_) to separate the dots and the j</span></span>
<span class=nb><span class=keyword>echo</span></span> <span class=k><span class=variable>${</span></span><span class=variable><span class=p>(j_._)my_array</span><span class=k>}</span></span><span class=k></span>
<span class=c><span class=comment># a.b.c.d</span></span>
</code></pre></div>
<h2>3. Magic tabbing</h2>
<h3>Event designators</h3>
<p>Let’s introduce the last member of our Zsh jargon family: after glob operators and qualifiers, modifiers and expansion flags, I give you <strong>event designators</strong>.</p>
<p>An event designator references one of the commands that we have previously entered. They always start with a bang <code>!</code>:</p>
<div class=highlight><pre><code class=language-bash data-lang=bash><span class=c><span class=comment># show the previous command</span></span>
<span class=nb><span class=keyword>echo</span> </span>a b c
!! <span class=c><span class=comment># instead of pressing <Enter>, press <Tab>, then press <Enter></span></span>
<span class=c><span class=comment># show two commands ago</span></span>
<span class=nb><span class=keyword>echo</span> </span>d e f
<span class=nb><span class=keyword>echo</span> </span>g h i
!-2 <span class=c><span class=comment># press <Tab>, then press <Enter></span></span>
</code></pre></div>
<p>Note: If you press <code><Enter></code> instead of <code><Tab></code>, the event designator will also get replaced, but you’ll still have to press <code><Enter></code> one more time to run it.</p>
<p>I find that these designators are not super useful by themselves because pressing the up arrow key is all we need to do to pull up the previous command (use <code>Control R</code> if you get tired of pressing). When they really come in handy is to add previous arguments to our current command.</p>
<div class=highlight><pre><code class=language-bash data-lang=bash><span class=c><span class=comment># add the last argument</span></span>
ls zsh_demo/data/asia/laos/population.txt
ls -l !!1 <span class=c><span class=comment># press <Tab>, then press <Enter></span></span>
<span class=c><span class=comment># add all the previous arguments</span></span>
<span class=nb><span class=keyword>echo</span> </span>a b c
print -l !!* <span class=c><span class=comment># press <Tab>, then press <Enter></span></span>
</code></pre></div>
<p>So, we reference previous arguments in two steps:</p>
<ol>
<li>Specify which command you are interested in
<ul>
<li>The previous command <code>!!</code> is the one you’ll use most often.</li>
<li>If you want to go back farther, use the minus sign <code>-</code> and a number: <code>!-2</code>, <code>!-3</code>.</li>
<li>You can also use the current command <code>!#</code></li>
</ul></li>
<li>Pick what arguments you want to reuse
<ul>
<li>To pick an argument from the previous command, just add a number <code>!!1</code>, <code>!!2</code>. Use <code>!!$</code> for the last argument.</li>
<li>To pick an argument from two or more commands ago, add a colon <code>:</code> before the number <code>!-2:1</code> (because<code>!-21</code> means something else).</li>
<li>If you want to reference all the arguments, use an asterisk <code>*</code> <code>!!*</code> <code>!-2:*</code>.</li>
<li>If you want skip all the arguments except the first one or two, add a number before the asterisk <code>!!2*</code>, <code>!-2:2*</code>.</li>
</ul></li>
</ol>
<p>Some useful examples</p>
<div class=highlight><pre><code class=language-bash data-lang=bash>mv zsh_demo/data/asia/laos/population.txt !<span class=comment>#1</span>
<span class=c><span class=comment># press <Tab></span></span>
<span class=c><span class=comment># now you can easily change the second argument</span></span>
<span class=c><span class=comment># (use Control W to delete every up to the first slash)</span></span>
ls zsh_demo/data/europe/malta/literacy.txt
awk <span class=s1><span class=string>'$1 > 3'</span></span> !<span class=err>$</span>
<span class=c><span class=comment># press <Tab></span></span>
<span class=c><span class=comment># !$ is a shortcut for !!$</span></span>
ls zsh_demo/*/*/nepal/literacy.txt
ls zsh_demo/*/*/malta/literacy.txt
ls -l !-2:1
<span class=c><span class=comment># press <Tab></span></span>
<span class=c><span class=comment># now you can see the details of the nepal file</span></span>
</code></pre></div>
<p>If you don’t believe there is such a thing as being <em>too productive</em>, check out the <a href=http://zsh.sourceforge.net/Doc/Release/Expansion.html#History-Expansion>history expansion section of the manual</a> for additional shortcuts.</p>
<p>Pressing <code><Tab></code> lets you expand not only old commands, but globs, variables (when they use the <code>${}</code> syntax), and even lazily-typed paths!</p>
<div class=highlight><pre><code class=language-bash data-lang=bash>ls zsh_demo/*/*/nepal/literacy.txt
<span class=c><span class=comment># press <Tab></span></span>
<span class=nv>my_var</span><span class=o>=</span><span class=s2><span class=string>"1 2 3"</span></span>
<span class=nb><span class=keyword>echo</span></span> <span class=k><span class=variable>${</span></span><span class=variable><span class=nv>my_var</span><span class=k>}</span></span><span class=k></span>
<span class=c><span class=comment># press <Tab></span></span>
ls z/d/a/l
<span class=c><span class=comment># press <Tab></span></span>
<span class=c><span class=comment># Mind blown!</span></span>
</code></pre></div>
<h2>Bonus tips</h2>
<p>There is a ton of stuff we haven’t covered, but I can point you to other people’s awesome tips. They are all totally worth it:</p>
<ul>
<li><p>Andrew Hays encourages us to <a href=http://www.andrewhays.net/2012/11/29/love-your-terminal.html>love our terminal</a> by installing the elegant <a href=http://ethanschoonover.com/solarized>Solarized color scheme</a>, and customizing our prompt. Follow his instructions and make all those hours in front of your terminal a more enjoyable experience.</p></li>
<li><p>Danilo Petrozzi over at Zsh Wiki shares a powerful alternative to global aliases. If you find yourself typing stuff like <code>| head | column -t | less -S</code> at the end of your commands, <a href=http://zshwiki.org/home/examples/zleiab>check out his method</a> to turn any sequence of characters into a convenient snippet.</p></li>
<li><p>Also at <a href=http://zshwiki.org/home/builtin/functions/zmv>Zsh Wiki</a>, learn about a way to rename multiple files by using the <code>zmv</code> command. It’s extremely convenient for replacing spaces with underscores, changing file extensions, and renaming files located in a nested folder structure. <strong>Always</strong> run <code>zmv</code> using the <code>-n</code> option once, so you are know what the command will actually do.</p></li>
<li><p>We used brace expansion multiple times in this tutorial, but we didn’t cover all of its features. Tomasz Muras wrote <a href=http://jmuras.com/blog/2012/brace-expansion-in-bash-and-zsh/>a nice post</a> about it.</p></li>
<li><p>I wasn’t able to find a blog post explaining <a href=http://zsh.sourceforge.net/Doc/Release/Parameters.html#Array-Parameters>associative arrays</a>, but I’ve found them incredibly useful to deal with sets of parameters. If there’s interest, I might go into them in a future post.</p></li>
</ul>
<h2>Parting thoughts</h2>
<p>I’ve you read this entire thing in one sitting your brain is probably melting. Don’t fret about remembering every little detail. What’s important is to realize that there are easy ways of doing a lot of things that you didn’t know could be done, and recalling where to go for additional information.</p>
<p>I have a file where I keep all my Zsh snippets. Every time I have to look up the syntax for a command, I open up the file and I write it down. I recommend you do the same. It’s the act of progressively improving this file that will help you achieve mastery.</p>
<p>Feel free to leave a comment if you’d like to share a tip.</p>
</div>
<ins class="adsbygoogle sf-hidden" style=display:inline-block;width:300px;height:250px;max-width:0px data-ad-client=ca-pub-9441199102473745 data-ad-slot=6108086510></ins>
<hr>
<div class=related>
<h3>Related posts</h3>
<ul>
<li><a href=http://reasoniamhere.com/2013/09/26/switching-between-long-and-wide-formats-in-r>Switching Between Long and Wide Formats in R</a></li>
<li><a href=http://reasoniamhere.com/2013/09/18/awk-gtf-how-to-analyze-a-transcriptome-like-a-pro-part-3>AWK GTF! How to Analyze a Transcriptome Like a Pro - Part 3</a></li>
<li><a href=http://reasoniamhere.com/2013/09/17/awk-gtf-how-to-analyze-a-transcriptome-like-a-pro-part-2>AWK GTF! How to Analyze a Transcriptome Like a Pro - Part 2</a></li>
<li><a href=http://reasoniamhere.com/2013/09/16/awk-gtf-how-to-analyze-a-transcriptome-like-a-pro-part-1>AWK GTF! How to Analyze a Transcriptome Like a Pro - Part 1</a></li>
</ul>
</div>
<hr>
<ul class="tag_box inline">
<li><i class=icon-tags></i></li>
<li><a href=http://reasoniamhere.com/tags.html#zsh-ref>zsh <span>1</span></a></li>
<li><a href=http://reasoniamhere.com/tags.html#tutorial-ref>tutorial <span>5</span></a></li>
</ul>
</div>
</div>
</div>
<footer>
<p class=copyright>© 2015 Nacho Caballero</p>
<p>
<a href=http://feeds.feedburner.com/TheReasonIAmHere>RSS</a> •
<a href=https://twitter.com/nachocaballero>Twitter</a> •
<a href=http://reasoniamhere.com/about.html>About</a>
</p>
</footer>
</div>
<script>document.currentScript.remove();!function(){"use strict";(t=>{const n="singlefile-infobar",e="",o="",i="SingleFile",A="single-file-ui-element",r="\n\t.infobar {\n\t\tbackground-color: #737373;\n\t\tcolor: white;\n\t\tdisplay: flex;\n\t\tposition: fixed;\n\t\ttop: 16px;\n\t\tright: 16px;\n\t\theight: auto;\n\t\twidth: auto;\n\t\tmin-height: 24px;\n\t\tmin-width: 24px;\n\t\tbackground-position: center;\n\t\tbackground-repeat: no-repeat;\n\t\tz-index: 2147483647;\n\t\tmargin: 0 0 0 16px;\n\t\tbackground-image: url();\n\t\tborder-radius: 16px;\n\t\tuser-select: none;\n\t\t-moz-user-select: none;\n\t\topacity: .7;\n\t\tcursor: pointer;\n\t\tpadding-left: 0;\n\t\tpadding-right: 0;\n\t\tpadding-top: 0;\n\t\tpadding-bottom: 0;\n\t\tborder: 2px solid #eee;\n\t\tbackground-size: 70% 70%;\n\t\ttransition: all 250ms;\n\t\tfont-size: 13px;\n\t}\n\t.infobar:hover {\n\t\topacity: 1;\n\t}\n\t.infobar-open {\n\t\topacity: 1;\n\t\tbackground-color: #f9f9f9;\n\t\tcursor: auto;\n\t\tcolor: #2d2d2d;\n\t\tpadding-top: 2px;\n\t\tpadding-bottom: 2px;\n\t\tborder: 2px solid #878787;\n\t\tbackground-image: none;\n\t\tborder-radius: 8px;\n\t\tuser-select: initial;\n\t\t-moz-user-select: initial;\n\t}\n\t.infobar-close-button {\n\t\tdisplay: none;\n\t\topacity: .7;\n\t\tpadding-top: 4px;\n\t\tpadding-left: 8px;\n\t\tpadding-right: 8px;\n\t\tcursor: pointer;\n\t\ttransition: opacity 250ms;\n\t\theight: 16px;\n\t}\n\t.infobar-close-button:hover {\n\t\topacity: 1;\n\t}\n\t.infobar-content {\n\t\tdisplay: none;\n\t\tfont-family: Arial;\n\t\tfont-size: 14px;\n\t\tline-height: 22px;\n\t\tword-break: break-word;\n\t\twhite-space: pre-wrap;\n\t\tposition: relative;\n\t\ttop: 1px;\n\t\ttext-align: left;\n\t}\n\t.infobar-link {\n\t\tdisplay: none;\n\t\tpadding-left: 8px;\n\t\tpadding-right: 8px;\n\t\tline-height: 11px;\n\t\tcursor: pointer;\n\t\tuser-select: none;\n\t\toutline: 0;\n\t}\n\t.infobar-link-icon {\n\t\tpadding-top: 4px;\n\t\tpadding-left: 2px;\n\t\tcursor: pointer;\n\t\topacity: .7;\n\t\ttransition: opacity 250ms;\n\t\theight: 16px;\n\t}\n\t.infobar-link-icon:hover {\n\t\topacity: 1;\n\t}\n\t.infobar-open .infobar-close-button, .infobar-open .infobar-content, .infobar-open .infobar-link {\n\t\tdisplay: inline-block;\n\t}";let a=!0;const c=t.browser;async function s(){const t=document.evaluate("//comment()",document,null,XPathResult.FIRST_ORDERED_NODE_TYPE,null);let s=t&&t.singleNodeValue;if(s&&((p=s).nodeType==Node.COMMENT_NODE&&p.textContent.includes(i))){const t=s.textContent.split("\n"),[,,i,...p]=t,g=i.match(/^ url: (.*) $/),h=g&&g[1];if(h){let t;if(c&&c.runtime&&c.runtime.sendMessage)try{t=await c.runtime.sendMessage({method:"tabs.getOptions",url:h})}catch(n){t={displayInfobar:!0}}else t={displayInfobar:!0};t.displayInfobar&&await async function(t,i){let c=document.querySelector(n);if(!c){let s="";if(i.length){const t=i[0].split("saved date: ")[1];if(t&&i.shift(),i.length>1){let t=i[0].split("info: ")[1].trim();for(let n=1;n<i.length-1;n++)t+="\n"+i[n].trim();s=t.trim()}else s=t}s=s||"No info",c=d(n,document.body),c.className=A;const p=await async function(t){if(t.attachShadow)return t.attachShadow({mode:"open"});{a=!1;const n=d("iframe",t);return n.style.setProperty("background-color","transparent","important"),n.style.setProperty("position","fixed","important"),n.style.setProperty("top",0,"important"),n.style.setProperty("right",0,"important"),n.style.setProperty("width","44px","important"),n.style.setProperty("height","48px","important"),n.style.setProperty("z-index",2147483647,"important"),new Promise((t=>{n.contentDocument.body.style.setProperty("margin",0),n.onload=()=>t(n.contentDocument.body)}))}}(c),g=document.createElement("style");g.textContent=r,p.appendChild(g);const h=document.createElement("div");h.classList.add("infobar"),p.appendChild(h);const u=document.createElement("img");u.classList.add("infobar-close-button"),h.appendChild(u),u.src=o,u.onclick=t=>{0===t.button&&c.remove()};const m=document.createElement("span");h.appendChild(m),m.classList.add("infobar-content"),m.textContent=s;const f=document.createElement("a");f.classList.add("infobar-link"),h.appendChild(f),f.target="_blank",f.rel="noopener noreferrer",f.title="Open source URL: "+t,f.href=t;const b=document.createElement("img");b.classList.add("infobar-link-icon"),f.appendChild(b),b.src=e,l(h),document.addEventListener("click",(t=>{if(0===t.button){let n=t.target;for(;n&&n!=c;)n=n.parentElement;n!=c&&l(h)}}))}}(h,p)}}var p}function l(t){if(t.classList.remove("infobar-open"),t.onclick=e=>{if(0===e.button)return function(t){a||document.querySelector(n).childNodes[0].contentWindow.getSelection().removeAllRanges();if(t.classList.add("infobar-open"),t.onclick=null,t.onmouseout=null,!a){const e=document.querySelector(n).childNodes[0];e.style.setProperty("width","100vw","important"),e.style.setProperty("height","100vh","important"),e.style.setProperty("width",t.getBoundingClientRect().width+33+"px","important"),e.style.setProperty("height",t.getBoundingClientRect().height+21+"px","important")}}(t),!1},!a){const t=document.querySelector(n).childNodes[0];t.style.setProperty("width","44px","important"),t.style.setProperty("height","48px","important")}}function d(t,n){const e=document.createElement(t);return n.appendChild(e),Array.from(getComputedStyle(e)).forEach((t=>e.style.setProperty(t,"initial","important"))),e}t.window==t.top&&("loading"==document.readyState?document.addEventListener("DOMContentLoaded",s,!1):s())})("object"==typeof globalThis?globalThis:window)}();
</script>