zhangjian
2023-05-30 dabbcc356af21f9f2f88ac69ff07994e6e32e4fc
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
'use strict';
 
var test = require('tape');
var parse = require('../').parse;
 
test('single operators', function (t) {
    t.same(parse('beep | boop'), ['beep', { op: '|' }, 'boop']);
    t.same(parse('beep|boop'), ['beep', { op: '|' }, 'boop']);
    t.same(parse('beep \\| boop'), ['beep', '|', 'boop']);
    t.same(parse('beep "|boop"'), ['beep', '|boop']);
 
    t.same(parse('echo zing &'), ['echo', 'zing', { op: '&' }]);
    t.same(parse('echo zing&'), ['echo', 'zing', { op: '&' }]);
    t.same(parse('echo zing\\&'), ['echo', 'zing&']);
    t.same(parse('echo "zing\\&"'), ['echo', 'zing\\&']);
 
    t.same(parse('beep;boop'), ['beep', { op: ';' }, 'boop']);
    t.same(parse('(beep;boop)'), [
        { op: '(' }, 'beep', { op: ';' }, 'boop', { op: ')' }
    ]);
 
    t.same(parse('beep>boop'), ['beep', { op: '>' }, 'boop']);
    t.same(parse('beep 2>boop'), ['beep', '2', { op: '>' }, 'boop']);
    t.same(parse('beep<boop'), ['beep', { op: '<' }, 'boop']);
 
    t.end();
});
 
test('double operators', function (t) {
    t.same(parse('beep || boop'), ['beep', { op: '||' }, 'boop']);
    t.same(parse('beep||boop'), ['beep', { op: '||' }, 'boop']);
    t.same(parse('beep ||boop'), ['beep', { op: '||' }, 'boop']);
    t.same(parse('beep|| boop'), ['beep', { op: '||' }, 'boop']);
    t.same(parse('beep  ||   boop'), ['beep', { op: '||' }, 'boop']);
 
    t.same(parse('beep && boop'), ['beep', { op: '&&' }, 'boop']);
    t.same(
        parse('beep && boop || byte'),
        ['beep', { op: '&&' }, 'boop', { op: '||' }, 'byte']
    );
    t.same(
        parse('beep&&boop||byte'),
        ['beep', { op: '&&' }, 'boop', { op: '||' }, 'byte']
    );
    t.same(
        parse('beep\\&\\&boop||byte'),
        ['beep&&boop', { op: '||' }, 'byte']
    );
    t.same(
        parse('beep\\&&boop||byte'),
        ['beep&', { op: '&' }, 'boop', { op: '||' }, 'byte']
    );
    t.same(
        parse('beep;;boop|&byte>>blip'),
        ['beep', { op: ';;' }, 'boop', { op: '|&' }, 'byte', { op: '>>' }, 'blip']
    );
 
    t.same(parse('beep 2>&1'), ['beep', '2', { op: '>&' }, '1']);
 
    t.same(
        parse('beep<(boop)'),
        ['beep', { op: '<(' }, 'boop', { op: ')' }]
    );
    t.same(
        parse('beep<<(boop)'),
        ['beep', { op: '<' }, { op: '<(' }, 'boop', { op: ')' }]
    );
 
    t.end();
});
 
test('duplicating input file descriptors', function (t) {
    // duplicating stdout to file descriptor 3
    t.same(parse('beep 3<&1'), ['beep', '3', { op: '<&' }, '1']);
 
    // duplicating stdout to file descriptor 0, i.e. stdin
    t.same(parse('beep <&1'), ['beep', { op: '<&' }, '1']);
 
    // closes stdin
    t.same(parse('beep <&-'), ['beep', { op: '<&' }, '-']);
 
    t.end();
});
 
test('here strings', function (t) {
    t.same(parse('cat <<< "hello world"'), ['cat', { op: '<<<' }, 'hello world']);
    t.same(parse('cat <<< hello'), ['cat', { op: '<<<' }, 'hello']);
    t.same(parse('cat<<<hello'), ['cat', { op: '<<<' }, 'hello']);
    t.same(parse('cat<<<"hello world"'), ['cat', { op: '<<<' }, 'hello world']);
 
    t.end();
});
 
test('glob patterns', function (t) {
    t.same(
        parse('tap test/*.test.js'),
        ['tap', { op: 'glob', pattern: 'test/*.test.js' }]
    );
 
    t.same(parse('tap "test/*.test.js"'), ['tap', 'test/*.test.js']);
    t.end();
});