coder/examples/examples.go

225 lines
5.3 KiB
Go
Raw Normal View History

package examples
import (
"archive/tar"
"bytes"
"embed"
"io"
"io/fs"
"path"
"sync"
"github.com/gohugoio/hugo/parser/pageparser"
"golang.org/x/sync/singleflight"
"golang.org/x/xerrors"
"github.com/coder/coder/codersdk"
)
var (
//go:embed templates
files embed.FS
exampleBasePath = "https://github.com/coder/coder/tree/main/examples/templates/"
examples = make([]codersdk.TemplateExample, 0)
parseExamples sync.Once
archives = singleflight.Group{}
ErrNotFound = xerrors.New("example not found")
)
const rootDir = "templates"
// List returns all embedded examples.
func List() ([]codersdk.TemplateExample, error) {
var returnError error
parseExamples.Do(func() {
files, err := fs.Sub(files, rootDir)
if err != nil {
returnError = xerrors.Errorf("get example fs: %w", err)
}
dirs, err := fs.ReadDir(files, ".")
if err != nil {
returnError = xerrors.Errorf("read dir: %w", err)
return
}
2022-04-01 19:42:36 +00:00
for _, dir := range dirs {
if !dir.IsDir() {
continue
}
exampleID := dir.Name()
exampleURL := exampleBasePath + exampleID
// Each one of these is a example!
readme, err := fs.ReadFile(files, path.Join(dir.Name(), "README.md"))
if err != nil {
returnError = xerrors.Errorf("example %q does not contain README.md", exampleID)
return
}
2022-04-01 19:42:36 +00:00
frontMatter, err := pageparser.ParseFrontMatterAndContent(bytes.NewReader(readme))
if err != nil {
returnError = xerrors.Errorf("parse example %q front matter: %w", exampleID, err)
return
}
2022-04-01 19:42:36 +00:00
nameRaw, exists := frontMatter.FrontMatter["name"]
if !exists {
returnError = xerrors.Errorf("example %q front matter does not contain name", exampleID)
return
}
2022-04-01 19:42:36 +00:00
name, valid := nameRaw.(string)
if !valid {
returnError = xerrors.Errorf("example %q name isn't a string", exampleID)
return
}
2022-04-01 19:42:36 +00:00
descriptionRaw, exists := frontMatter.FrontMatter["description"]
if !exists {
returnError = xerrors.Errorf("example %q front matter does not contain name", exampleID)
return
}
2022-04-01 19:42:36 +00:00
description, valid := descriptionRaw.(string)
if !valid {
returnError = xerrors.Errorf("example %q description isn't a string", exampleID)
return
}
2022-04-01 19:42:36 +00:00
tags := []string{}
tagsRaw, exists := frontMatter.FrontMatter["tags"]
if exists {
tagsI, valid := tagsRaw.([]interface{})
if !valid {
returnError = xerrors.Errorf("example %q tags isn't a slice: type %T", exampleID, tagsRaw)
return
}
for _, tagI := range tagsI {
tag, valid := tagI.(string)
if !valid {
returnError = xerrors.Errorf("example %q tag isn't a string: type %T", exampleID, tagI)
return
}
tags = append(tags, tag)
}
}
var icon string
iconRaw, exists := frontMatter.FrontMatter["icon"]
if exists {
icon, valid = iconRaw.(string)
if !valid {
returnError = xerrors.Errorf("example %q icon isn't a string", exampleID)
return
}
}
examples = append(examples, codersdk.TemplateExample{
ID: exampleID,
URL: exampleURL,
Name: name,
Description: description,
Icon: icon,
Tags: tags,
Markdown: string(frontMatter.Content),
})
}
})
return examples, returnError
}
// Archive returns a tar by example ID.
func Archive(exampleID string) ([]byte, error) {
rawData, err, _ := archives.Do(exampleID, func() (interface{}, error) {
examples, err := List()
if err != nil {
2022-04-01 19:42:36 +00:00
return nil, xerrors.Errorf("list: %w", err)
}
2022-04-01 19:42:36 +00:00
var selected codersdk.TemplateExample
for _, example := range examples {
if example.ID != exampleID {
continue
}
selected = example
break
}
2022-04-01 19:42:36 +00:00
if selected.ID == "" {
return nil, xerrors.Errorf("example with id %q not found: %w", exampleID, ErrNotFound)
}
exampleFiles, err := fs.Sub(files, path.Join(rootDir, exampleID))
if err != nil {
return nil, xerrors.Errorf("get example fs: %w", err)
}
var buffer bytes.Buffer
tarWriter := tar.NewWriter(&buffer)
err = fs.WalkDir(exampleFiles, ".", func(path string, entry fs.DirEntry, err error) error {
if err != nil {
return err
}
2022-04-01 19:42:36 +00:00
info, err := entry.Info()
if err != nil {
return xerrors.Errorf("stat file: %w", err)
}
2022-04-01 19:42:36 +00:00
header, err := tar.FileInfoHeader(info, entry.Name())
if err != nil {
return xerrors.Errorf("get file header: %w", err)
}
header.Mode = 0644
2022-04-01 19:42:36 +00:00
if entry.IsDir() {
header.Name = path + "/"
err = tarWriter.WriteHeader(header)
if err != nil {
return xerrors.Errorf("write file: %w", err)
}
} else {
header.Name = path
file, err := exampleFiles.Open(path)
if err != nil {
return xerrors.Errorf("open file %s: %w", path, err)
}
defer file.Close()
err = tarWriter.WriteHeader(header)
if err != nil {
return xerrors.Errorf("write file: %w", err)
}
_, err = io.Copy(tarWriter, file)
if err != nil {
return xerrors.Errorf("write: %w", err)
}
}
return nil
})
if err != nil {
return nil, xerrors.Errorf("walk example directory: %w", err)
}
err = tarWriter.Close()
if err != nil {
return nil, xerrors.Errorf("close archive: %w", err)
}
2022-04-01 19:42:36 +00:00
return buffer.Bytes(), nil
})
if err != nil {
return nil, err
}
data, valid := rawData.([]byte)
if !valid {
panic("dev error: data must be a byte slice")
}
return data, nil
}