-
Notifications
You must be signed in to change notification settings - Fork 58
/
shasum.ps1
104 lines (83 loc) · 3.03 KB
/
shasum.ps1
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
Set-StrictMode -Off;
. "$psscriptroot\getopt.ps1"
function usage {
"Usage: shasum [OPTION] [FILE]...
or: shasum [OPTION] --check [FILE]
Print or check SHA checksums.
With no FILE, or when FILE is -, read standard input.
-a, --algorithm 1 (default), 256, 384, 512
-b, --binary read files in binary mode (default)
-c, --check check SHA sums against given list
-t, --text read files in text mode
-p, --portable read files in portable mode
produces same digest on Windows/Unix/Mac
The following two options are useful only when verifying checksums:
-s, --status don't output anything, status code shows success
-w, --warn warn about improperly formatted SHA checksum lines
-h, --help display this help and exit
-v, --version output version information and exit
The sums are computed as described in FIPS PUB 180-2. When checking,
the input should be a former output of this program. The default mode
is to print a line with checksum, a character indicating type (`*'
for binary, `?' for portable, ` ' for text), and name for each FILE."
}
$algs = @(1,256,384,512)
function compute_hash($file, $algname) {
$alg = [system.security.cryptography.hashalgorithm]::create($algname)
$fs = [system.io.file]::openread($file)
try {
$hexbytes = $alg.computehash($fs) | % { $_.tostring('x2') }
[string]::join('', $hexbytes)
} finally {
$fs.dispose()
$alg.dispose()
}
}
function write_hash($file, $alg, $mode) {
if($file -match '\*') { "shasum: $file`: invalid argument"; return }
if(!(test-path $file -pathtype leaf)) { "shasum: $file`: no such file"; return }
$hash = compute_hash (resolve-path $file) "SHA$alg"
$mode_indicator = switch($mode) {
'binary' { '*' }
'text' { ' '}
'portable' { '?' }
}
"$hash $mode_indicator$file"
}
function verify($checkfile) {
if(!(test-path $checkfile -pathtype leaf)) { "shasum: $file`: no such file"; return }
$len2alg = @{ 40 = 1; 56 = 224; 64 = 256; 96 = 384; 128 = 512 }
$lines = gc $checkfile
$lines | % {
if($_ -match '([^ ]+) (.)(.*)') {
$hash, $mode, $file = $matches[1..3]
$alg = $len2alg[$hash.length]
if($algs -notcontains $alg) {
"$file`: FAILED: SHA-$alg not supported"; return
}
if(!(test-path $file -pathtype leaf)) {
"$file`: FAILED: no such file"; return
}
$match = $hash -eq (compute_hash (resolve-path $file) "SHA$alg")
"$file`: $(if($match) { 'OK' } else { 'FAILED' })"
}
}
}
$opt, $files, $err = getopt $args 'a:bcpth' @('algorithm=','binary','check','text','portable','help')
if($err) { "shasum: $err"; exit 1 }
if($opt.h -or $opt.help) { usage; exit 0 }
if(!$files) { "shasum: file is required"; exit 1 }
$alg = $opt.algorithm;
if(!$alg) { $alg = $opt.a }
if(!$alg) { $alg = 1 }
if($algs -notcontains $alg) { "shasum: invalid algorithm"; exit 1 }
$mode = 'binary'
if($opt.t -or $opt.text) { $mode = 'text' }
if($opt.p -or $opt.portable) { $mode = 'portable' }
$check = $opt.c -or $opt.check
if($check) {
verify @($files)[0]
} else {
$files | % { write_hash $_ $alg $mode }
}
exit 0