-
Notifications
You must be signed in to change notification settings - Fork 43
/
Copy pathdocument.go
159 lines (136 loc) · 3.68 KB
/
document.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
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
// A quick and easy way to review the documentation in the cloud-platform repository.
// You can execute this using the Go binary, like so:
// go run document.go --dir sources/docs
// To correct the dates simply add a --fix to the end, like so:
// go run document.go --dir sources/docs --fix
package main
import (
"bytes"
"flag"
"fmt"
"io/ioutil"
"log"
"os"
"path/filepath"
"strings"
"time"
)
func main() {
var (
dir string
fix bool
)
flag.StringVar(&dir, "dir", ".", "The directory to search for documents")
flag.BoolVar(&fix, "fix", false, "Set the out of date documents to todays date")
flag.Parse()
threeMonthsAgo := time.Now().AddDate(0, -3, 0).Format("2006-01-02")
// Grab all documents needed to review.
docs, err := allDocuments(dir)
if err != nil {
log.Fatalln(err)
}
for _, doc := range docs {
needsReview, err := needsReview(doc, threeMonthsAgo)
if err != nil {
log.Printf("Unable to check if document: %s needs review %s\n", doc, err)
}
if needsReview {
if fix {
err := setLastReviewedOn(doc)
if err != nil {
log.Println(err)
}
continue
}
fmt.Println(doc)
}
}
}
func setLastReviewedOn(filePath string) error {
b, err := ioutil.ReadFile(filePath)
if err != nil {
return err
}
lastReviewedOn, err := lastReviewedOn(filePath)
if err != nil {
return fmt.Errorf("unable to get last_reviewed_on: %w", err)
}
// We have to crowbar a space to make the date valid.
todaysDate := " " + time.Now().Format("2006-01-02")
o := bytes.Replace(b, []byte(lastReviewedOn), []byte(todaysDate), 1)
err = ioutil.WriteFile(filePath, o, 0o644)
if err != nil {
return fmt.Errorf("unable to write file: %w", err)
}
return nil
}
func parseTime(timeString string) (time.Time, error) {
format := "2006-01-02"
return time.Parse(format, strings.TrimSpace(timeString))
}
func needsReview(filePath, threeMonthsAgo string) (bool, error) {
lastReviewedOn, err := lastReviewedOn(filePath)
if err != nil {
return false, fmt.Errorf("unable to get last_reviewed_on: %w", err)
}
// Compare the last_reviewed_on date to the threeMonthsAgo date.
lastReviewedOnTime, err := parseTime(lastReviewedOn)
if err != nil {
return false, fmt.Errorf("unable to parse last_reviewed_on: %s, %w", lastReviewedOn, err)
}
threeMonthsAgoTime, err := parseTime(threeMonthsAgo)
if err != nil {
return false, fmt.Errorf("unable to parse time: %s", threeMonthsAgo)
}
if lastReviewedOnTime.Before(threeMonthsAgoTime) {
return true, nil
}
return false, nil
}
func lastReviewedOn(filePath string) (string, error) {
// Grab the last_reviewed_on date from the document.
// read the whole file at once
b, err := os.ReadFile(filePath)
if err != nil {
return "", fmt.Errorf("unable to read file: %w", err)
}
lines := strings.Split(string(b), "\n")
for _, line := range lines {
if strings.Contains(line, "last_reviewed_on") {
return strings.Split(line, ":")[1], nil
}
}
// find the last_reviewed_on date
return "", fmt.Errorf("unable to find last_reviewed_on in file: %s", filePath)
}
func contains(slice []string, item string) bool {
set := make(map[string]struct{}, len(slice))
for _, s := range slice {
set[s] = struct{}{}
if _, ok := set[item]; ok {
return true
}
}
return false
}
func allDocuments(dir string) ([]string, error) {
ignored := []string{
"incident-log.html.md.erb",
"index.html.md.erb",
"custom.erb",
}
var documents []string
err := filepath.Walk(dir, func(path string, info os.FileInfo, err error) error {
if err != nil {
return err
}
if filepath.Ext(path) == ".erb" && !contains(ignored, info.Name()) {
documents = append(documents, path)
}
return nil
})
if err != nil {
return nil, err
}
return documents, nil
}