-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathpkcs7.go
92 lines (76 loc) · 2.88 KB
/
pkcs7.go
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
// This package implements PKCS#7 padding to byte slices. Padding works by
// calculating the amount of needed padding and repeating that number to fill in
// the rest of the slice up to the block size. This way, in order to unpad the
// slice, you check the value held in the last slot and then remove that many
// bytes from the end.
//
// By defniition, PKCS#7 only padds for block sizes between 1 and 255 inclusive.
// If the supplied byte slice is a multiple of the block size, N, an extra N
// amount of bytes is appended all of value N.
//
// Please review the tests for this package for examples.
package pkcs7
import (
"bytes"
"errors"
)
var (
ErrInvalidBlockSize = errors.New("pkcs7: block size must be between 1 and 255 inclusive")
ErrEmptySlice = errors.New("pkcs7: source must not be empty slice")
ErrInvalidPadding = errors.New("pkcs7: invalid padding")
)
// Pad takes a source byte slice and a block size. It will determine the needed
// amount of padding, n, and appends byte(n) to the source n times.
//
// Example Input: Block Size 8, Source {0xDE, 0xAD, 0xBE, 0xEF}
//
// Expected Output: {0xDE, 0xAD, 0xBE, 0xEF, 0x04, 0x04, 0x04, 0x04}
//
func Pad(src []byte, blockSize int) ([]byte, error) {
// Only allow 1-255 sized blocks as per standard.
if blockSize < 1 || blockSize > 255 {
return nil, ErrInvalidBlockSize
}
// Calculate length of needed padding by taking the goal block size and
// subtracting the overflow of the source.
padLen := blockSize - len(src)%blockSize
// Make a byte slice containing the byte to be repeated.
padding := []byte{byte(padLen)}
// repeat that byte padLen times
padding = bytes.Repeat(padding, padLen)
// Append the padding to the src.
return append(src, padding...), nil
}
// Unpad takes a source byte slice and will remove any padding added according
// to PKCS#7 specifications. An error is returned for invalid padding.
func Unpad(src []byte) ([]byte, error) {
length := len(src)
// If the source is empty it's already invalid.
if length <= 0 {
return nil, ErrEmptySlice
}
// Get the last byte so we know how many bytes to take off the end.
padLen := int(src[length-1])
// If the last byte is 0x00, we have invalid padding. We try to fuzz a bit
// the error message, sending the same one as when the padding is incorrect.
if padLen == 0x00 {
return nil, ErrInvalidPadding
}
// If the last byte is more than the total length, this is invalid.
if padLen > length {
return nil, ErrInvalidPadding
}
// Get original source length assumed based on last byte.
origLen := length - padLen
// Get all the padding so we can check it's actually padding and not just an
// invalid last byte.
padding := src[origLen:]
for i := 0; i < padLen; i++ {
// Make sure all bytes match.
if padding[i] != byte(padLen) {
return nil, ErrInvalidPadding
}
}
// Return the source bytes up to the start of the padding.
return src[:origLen], nil
}