Skip to content

Commit e965fd5

Browse files
committed
Fix a lot of bugs
1 parent 8476e82 commit e965fd5

9 files changed

+225
-146
lines changed

NOTES.md

+21
Original file line numberDiff line numberDiff line change
@@ -17,6 +17,8 @@ CGO_ENABLED=1 go build --tags "fts5" -o "dist/bock-$(uname)-$(uname -m)" .
1717
### TODO
1818

1919
* [x] FIX THE NAVIGATION FFS
20+
* [ ] FIX THE PATH GENERATION PROBLEM FFS
21+
* [ ] FIX FOLDER generation
2022
* [x] Breadcrumbs > Revision
2123
* [ ] Table of Contents
2224
* [ ] Recent Changes (Global)
@@ -33,6 +35,9 @@ CGO_ENABLED=1 go build --tags "fts5" -o "dist/bock-$(uname)-$(uname -m)" .
3335
* [x] 404 Page
3436
* [ ] Template argument
3537
* [ ] Revisions argument
38+
* [ ] Better, less buggy tree
39+
* [ ] Gist of recursive tree generation!
40+
- [ ] Fix issue with apostrophes 🤦‍♀️
3641
* [ ] Use `context` in lieu of `config` struct? What are the dis/advantages?
3742
* [ ] [Markdown highlight in Raw view](https://www.zupzup.org/go-markdown-syntax-highlight-chroma/)
3843
* [ ] [Filtering logs with filename is very slow in `go-git`](https://github.com/go-git/go-git/issues/137)```
@@ -328,3 +333,19 @@ func main() {
328333
fmt.Println(string(s))
329334
}
330335
```
336+
337+
### Pongo2 Custom Filter Sample
338+
339+
```golang
340+
var _ = pongo2.RegisterFilter("round", func(in, param *pongo2.Value) (out *pongo2.Value, err *pongo2.Error) {
341+
var rounded *pongo2.Value
342+
343+
if s, err := strconv.ParseFloat(in.String(), 32); err == nil {
344+
rounded = pongo2.AsSafeValue(math.Round(s))
345+
} else {
346+
return pongo2.AsSafeValue("!_COULD_NOT_ROUND_VALUE"), &pongo2.Error{OrigError: err}
347+
}
348+
349+
return rounded, nil
350+
})
351+
```

README.md

+5-1
Original file line numberDiff line numberDiff line change
@@ -17,6 +17,10 @@ git clone https://github.com/afreeorange/bockgo.git
1717
go run --tags "fts5" . -a /path/to/repo -o /path/to/output -r -j
1818
```
1919

20+
## Terminology et al
21+
22+
An "Entity" is either an "Article" (a Markdown file somewhere in your article repository) or a "Folder" (which is exactly what you think it is). A "Revision" is a `git` commit that modifies an Article.
23+
2024
* You can organize your articles into folders.
2125
* You cannot use `raw`, `revisions`, `random`, and `archive` as folder names.
2226
* You can place static assets in `__assets` in your article repository. You can reference all assets in there in your Markdown files prefixed with `/assets` (e.g. `__assets/some-file.jpg` → `/assets/some-file.jpg`).
@@ -27,7 +31,7 @@ The command will generate the following:
2731
* Each article's revision rendered as Raw Source, HTML, and JSON
2832
* A listing of all revisions for each article (if applicable)
2933
* Each folder's structure in HTML and JSON ([example](https://wiki.nikhil.io/Food/))
30-
* An archive page that lets you search your articles
34+
* An archive page that lets you search your articles thanks to SQLite and [SQL.js](https://github.com/sql-js/sql.js/)
3135
* A Homepage (if it doesn't exist as `Home.md`) at `/Home`
3236
* A page that redirects to some random article at `/random`
3337
* An index page that redirects to `/Home`

VERSION

+1-1
Original file line numberDiff line numberDiff line change
@@ -1 +1 @@
1-
4.0.0-beta.18
1+
4.0.0-beta.19

constants.go

+5-21
Original file line numberDiff line numberDiff line change
@@ -31,20 +31,16 @@ const (
3131
)
3232

3333
// Things to ignore when walking the article repository. NOTE: In Golang, only
34-
// primitive types (like `int`, `string`, etc) can be constants.
35-
var IGNORED_FOLDERS_REGEX = regexp.MustCompile(strings.Join([]string{
34+
// primitive types (like `int`, `string`, etc) can be constants. NOTE: Folders
35+
// beginning with `.` are automatically excluded in the function that uses this
36+
// pattern.
37+
var IGNORED_ENTITIES_REGEX = regexp.MustCompile(strings.Join([]string{
3638
"__assets",
37-
"_assets",
3839
"css",
3940
"img",
4041
"js",
42+
"Home.md", // This is dealt with separately
4143
"node_modules",
42-
// NOTE: Folders beginning with `.` are automatically excluded
43-
}, "|"))
44-
45-
// Here since we process the Homepage separately
46-
var IGNORED_FILES_REGEX = regexp.MustCompile(strings.Join([]string{
47-
"Lol.md",
4844
}, "|"))
4945

5046
// We use Goldmark as the Markdown converter. Configure it here.
@@ -72,15 +68,3 @@ var markdown = goldmark.New(
7268
var templatesContent embed.FS
7369
var pongoLoader = pongo2.NewFSLoader(templatesContent)
7470
var templateSet = pongo2.NewSet("template", pongoLoader)
75-
76-
// var _ = pongo2.RegisterFilter("round", func(in, param *pongo2.Value) (out *pongo2.Value, err *pongo2.Error) {
77-
// var rounded *pongo2.Value
78-
79-
// if s, err := strconv.ParseFloat(in.String(), 32); err == nil {
80-
// rounded = pongo2.AsSafeValue(math.Round(s))
81-
// } else {
82-
// return pongo2.AsSafeValue("!_COULD_NOT_ROUND_VALUE"), &pongo2.Error{OrigError: err}
83-
// }
84-
85-
// return rounded, nil
86-
// })

helpers.go

+72-17
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,7 @@ import (
77
"fmt"
88
"io/fs"
99
"os"
10+
"path"
1011
"path/filepath"
1112
"sort"
1213
"strings"
@@ -153,7 +154,7 @@ func getEntityInfo(config *BockConfig, info fs.FileInfo, path string) *Entity {
153154
IsFolder: info.IsDir(),
154155
Modified: info.ModTime(),
155156
Name: info.Name(),
156-
Path: path,
157+
path: path,
157158
RelativePath: makeRelativePath(path, config.articleRoot),
158159
SizeInBytes: info.Size(),
159160
Title: removeExtensionFrom(info.Name()),
@@ -176,7 +177,7 @@ func findChildWithName(children *[]Entity, name string) int {
176177
return -1
177178
}
178179

179-
func makeEntityTree(listOfArticles *[]Entity) []Entity {
180+
func makeEntityTree(config *BockConfig) []Entity {
180181
tree := []Entity{}
181182

182183
// Bootstrap: create adn append the root entity (a folder)
@@ -189,12 +190,12 @@ func makeEntityTree(listOfArticles *[]Entity) []Entity {
189190
SizeInBytes: 0,
190191
Title: "Root",
191192
URI: "/ROOT",
192-
Path: ".",
193+
path: ".",
193194
})
194195

195196
// These loops took me an embarrassingly LONG while to write :/
196197

197-
for _, article := range *listOfArticles {
198+
for _, article := range *config.listOfArticles {
198199
pathFragments := strings.Split(article.RelativePath, "/")
199200

200201
// Use this to build the URI. Reset with each iteration.
@@ -227,8 +228,8 @@ func makeEntityTree(listOfArticles *[]Entity) []Entity {
227228
RelativePath: strings.TrimPrefix(uri, "/"),
228229
SizeInBytes: 0,
229230
Title: fragment,
230-
URI: uri,
231-
Path: article.Path,
231+
URI: makeURI(uri, config.articleRoot),
232+
path: article.path,
232233
})
233234
}
234235
}
@@ -243,26 +244,80 @@ func makeEntityTree(listOfArticles *[]Entity) []Entity {
243244
return tree
244245
}
245246

246-
func makeListOfArticles(config *BockConfig) ([]Entity, error) {
247-
list := []Entity{}
247+
func uniqueStringsInList(list []string) []string {
248+
var uniqueList []string
249+
tempMap := make(map[string]bool)
248250

249-
walkFunction := func(path string, entityInfo os.FileInfo, err error) error {
250-
relativePath := makeRelativePath(path, config.articleRoot)
251+
for _, e := range list {
252+
if _, ok := tempMap[e]; !ok {
253+
tempMap[e] = true
254+
uniqueList = append(uniqueList, e)
255+
}
256+
}
257+
258+
return uniqueList
259+
}
260+
261+
func makeListOfEntities(config *BockConfig) (
262+
listOfArticles []Entity,
263+
listOfFolders []string,
264+
err error,
265+
) {
266+
walkFunction := func(entityPath string, entityInfo os.FileInfo, walkErr error) error {
267+
relativePath := makeRelativePath(entityPath, config.articleRoot)
251268

252269
isValidArticle := (!entityInfo.IsDir() &&
253-
!IGNORED_FILES_REGEX.MatchString(path) &&
270+
!IGNORED_ENTITIES_REGEX.MatchString(entityPath) &&
254271
!strings.HasPrefix(relativePath, ".") &&
255-
filepath.Ext(path) == ".md")
272+
filepath.Ext(entityPath) == ".md")
256273

257274
if isValidArticle {
258-
list = append(list, *getEntityInfo(config, entityInfo, path))
275+
listOfArticles = append(
276+
listOfArticles,
277+
*getEntityInfo(config, entityInfo, entityPath),
278+
)
279+
280+
folderPath := path.Dir(entityPath)
281+
282+
/*
283+
For example,
284+
285+
/article/root/sso-react
286+
/article/root/sso-react/build/refresh
287+
/article/root/sso-react/public/refresh
288+
/article/root/sso-react/src/i18n
289+
290+
should be
291+
292+
/article/root/sso-react
293+
/article/root/sso-react/build
294+
/article/root/sso-react/build/refresh
295+
/article/root/sso-react/public
296+
/article/root/sso-react/public/refresh
297+
/article/root/sso-react/src
298+
/article/root/sso-react/src/i18n
299+
300+
That's what we're doing here.
301+
*/
302+
folderSplits := strings.Split(makeRelativePath(folderPath, config.articleRoot), "/")
303+
p := ""
304+
if len(folderSplits) > 1 {
305+
for _, s := range folderSplits {
306+
p += "/" + s
307+
listOfFolders = append(listOfFolders, config.articleRoot+p)
308+
}
309+
} else {
310+
listOfFolders = append(listOfFolders, folderPath)
311+
}
259312
}
260313

261314
return nil
262315
}
263316

264-
// Make a list of articles and return it
265-
// TODO: Sort?
266-
err := filepath.Walk(config.articleRoot, walkFunction)
267-
return list, err
317+
// It strikes me that error-handling in Go is a bit strange... looks like
318+
// things can just fall through.
319+
err = filepath.Walk(config.articleRoot, walkFunction)
320+
listOfFolders = uniqueStringsInList(listOfFolders)
321+
322+
return listOfArticles, listOfFolders, err
268323
}

main.go

+25-22
Original file line numberDiff line numberDiff line change
@@ -17,23 +17,23 @@ import (
1717
)
1818

1919
func main() {
20-
var versionInfo bool
2120
var articleRoot string
22-
var outputFolder string
2321
var generateJSON bool
2422
var generateRaw bool
25-
var useOnDiskFS bool
23+
var generateRevisions bool
2624
var minifyOutput bool
27-
var createRevisions bool
25+
var outputFolder string
26+
var useOnDiskFS bool
27+
var versionInfo bool
2828

29-
flag.BoolVar(&versionInfo, "v", false, "Version info")
3029
flag.StringVar(&articleRoot, "a", "", "Article root")
31-
flag.StringVar(&outputFolder, "o", "", "Output folder")
3230
flag.BoolVar(&generateJSON, "j", false, "Create JSON source files")
3331
flag.BoolVar(&generateRaw, "r", false, "Create Raw markdown source files")
34-
flag.BoolVar(&useOnDiskFS, "d", false, "Use on-disk filesystem to clone article repository (slower; cloned to memory by default)")
32+
flag.BoolVar(&generateRevisions, "R", true, "Create article revisions based on git history (default: true)")
3533
flag.BoolVar(&minifyOutput, "m", false, "Minify all output (HTML, JS, CSS)")
36-
flag.BoolVar(&createRevisions, "R", true, "Create article revisions based on git history (default: true)")
34+
flag.StringVar(&outputFolder, "o", "", "Output folder")
35+
flag.BoolVar(&useOnDiskFS, "d", false, "Use on-disk filesystem to clone article repository (slower; cloned to memory by default)")
36+
flag.BoolVar(&versionInfo, "v", false, "Version info")
3737

3838
flag.Parse()
3939

@@ -100,29 +100,32 @@ func main() {
100100
database: nil,
101101
outputFolder: outputFolder,
102102
meta: Meta{
103-
Architecture: runtime.GOARCH,
104-
ArticleCount: 0,
105-
BuildDate: time.Now().UTC(),
106-
CPUCount: runtime.NumCPU(),
107-
GenerateJSON: generateJSON,
108-
GenerateRaw: generateRaw,
109-
GenerationTime: 0,
110-
MemoryInGB: int(v.Total / (1024 * 1024 * 1024)),
111-
Platform: runtime.GOOS,
112-
RevisionCount: 0,
103+
Architecture: runtime.GOARCH,
104+
ArticleCount: 0,
105+
BuildDate: time.Now().UTC(),
106+
CPUCount: runtime.NumCPU(),
107+
GenerateJSON: generateJSON,
108+
GenerateRaw: generateRaw,
109+
GenerateRevisions: generateRevisions,
110+
GenerationTime: 0,
111+
MemoryInGB: int(v.Total / (1024 * 1024 * 1024)),
112+
Platform: runtime.GOOS,
113+
RevisionCount: 0,
113114
},
114115
started: time.Now(),
115116
repository: repository,
116117
workTreeStatus: &status,
117118
}
118119

119-
// Make a flat list of absolute article paths
120-
listOfArticles, _ := makeListOfArticles(&config)
120+
// Make a flat list of absolute article paths. Use these to build the entity
121+
// tree. We do this to prevent unnecessary and empty folders from being
122+
// created.
123+
listOfArticles, _, _ := makeListOfEntities(&config)
121124
config.listOfArticles = &listOfArticles
122-
123125
fmt.Println("Found", len(*config.listOfArticles), "articles")
124126

125-
entityTree := makeEntityTree(config.listOfArticles)
127+
// Make a tree of entities: articles and folders
128+
entityTree := makeEntityTree(&config)
126129
config.entityTree = &entityTree
127130

128131
// Database setup

template/article.njk

+1-1
Original file line numberDiff line numberDiff line change
@@ -17,5 +17,5 @@
1717
{% endblock footerElements %}
1818

1919
{% block statistics %}
20-
There are {{ meta.ArticleCount }} articles and {{ meta.RevisionCount }} revisions in this wiki. On {{ meta.BuildDate | date:"Monday, 2 January 2006 at 15:04 MST" }}, it took {{ meta.GenerationTimeRounded }} to generate it on a {{ meta.CPUCount }}-core {{ meta.Platform }}/{{ meta.Architecture }} system with {{ meta.MemoryInGB }}GiB RAM
20+
There are {{ meta.ArticleCount }} articles and {{ meta.RevisionCount }} revisions in this wiki. It took {{ meta.GenerationTimeRounded }} to generate it on a {{ meta.CPUCount }}-core {{ meta.Platform }}/{{ meta.Architecture }} system with {{ meta.MemoryInGB }}GiB RAM on {{ meta.BuildDate | date:"Monday, 2 January 2006 at 15:04 MST" }}
2121
{% endblock statistics %}

types.go

+3-1
Original file line numberDiff line numberDiff line change
@@ -64,14 +64,16 @@ type Entity struct {
6464
Title string `json:"title"`
6565
URI string `json:"uri"`
6666

67-
Path string `json:"path"`
67+
// You do NOT want to make this public!
68+
path string
6869
}
6970

7071
type Meta struct {
7172
Architecture string `json:"architecture"`
7273
ArticleCount int `json:"articleCount"`
7374
BuildDate time.Time `json:"buildTime"`
7475
CPUCount int `json:"cpuCount"`
76+
GenerateRevisions bool `json:"generateRevisions"`
7577
GenerateJSON bool `json:"generateJSON"`
7678
GenerateRaw bool `json:"generateRaw"`
7779
GenerationTime time.Duration `json:"generationTime"`

0 commit comments

Comments
 (0)