mark2web/main.go

337 lines
8.9 KiB
Go
Raw Normal View History

2019-02-10 13:49:27 +01:00
package main
import (
2019-02-11 15:00:27 +01:00
"flag"
2019-02-10 13:49:27 +01:00
"io/ioutil"
2019-02-11 15:00:27 +01:00
"os"
2019-02-11 15:25:48 +01:00
"regexp"
2019-02-11 15:00:27 +01:00
"strings"
2019-02-10 13:49:27 +01:00
2019-02-11 15:00:27 +01:00
"github.com/davecgh/go-spew/spew"
"github.com/op/go-logging"
"gopkg.in/yaml.v2"
2019-02-10 13:49:27 +01:00
)
2019-02-11 15:00:27 +01:00
var log = logging.MustGetLogger("myLogger")
var inDir *string
var outDir *string
// GlobalConfig is config which is used only once in root dir
type GlobalConfig struct {
Webserver struct {
Type *string `yaml:"type"`
} `yaml:"webserver"`
Assets struct {
Path *string `yaml:"path"`
DeployType *string `yaml:"deployType"`
FixTemplate *bool `yaml:"fixTemplate"`
} `yaml:"assets"`
}
var globalConfig = new(GlobalConfig)
// PathConfig of subdir
type PathConfig struct {
Title *string `yaml:"title"`
Navname *string `yaml:"navname"`
GoTo *string `yaml:"goTo"`
InputFile *string `yaml:"inputFile"`
OutputFile *string `yaml:"outputFile"`
Meta struct {
Title *string `yaml:"title"`
Description *string `yaml:"description"`
Keywords *string `yaml:"keywords"`
} `yaml:"meta"`
Path struct {
Strip *string `yaml:"strip"`
IgnoreForNav *string `yaml:"ignoreForNav"`
} `yaml:"path"`
2019-02-12 10:21:51 +01:00
Filename struct {
Strip *string `yaml:"strip"`
Ignore *string `yaml:"ignore"`
OutputExtension *string `yaml:"outputExtension"`
} `yaml:"filename"`
2019-02-11 15:00:27 +01:00
}
// PathConfigTree is complete config tree of content dir
type PathConfigTree struct {
InputPath string
OutputPath string
2019-02-12 10:21:51 +01:00
InputFiles []string
2019-02-11 15:00:27 +01:00
Config *PathConfig
Sub []*PathConfigTree
}
func mergeConfig(mergeInto *PathConfig, mergeFrom *PathConfig) {
if mergeInto.InputFile == nil {
mergeInto.InputFile = mergeFrom.InputFile
}
if mergeInto.OutputFile == nil {
mergeInto.OutputFile = mergeFrom.OutputFile
}
if mergeInto.Meta.Title == nil {
mergeInto.Meta.Title = mergeFrom.Meta.Title
}
if mergeInto.Meta.Description == nil {
mergeInto.Meta.Description = mergeFrom.Meta.Description
}
if mergeInto.Meta.Keywords == nil {
mergeInto.Meta.Keywords = mergeFrom.Meta.Keywords
}
if mergeInto.Path.IgnoreForNav == nil {
mergeInto.Path.IgnoreForNav = mergeFrom.Path.IgnoreForNav
}
if mergeInto.Path.Strip == nil {
mergeInto.Path.Strip = mergeFrom.Path.Strip
}
}
var contentConfig = new(PathConfigTree)
2019-02-12 10:21:51 +01:00
func readContentDir(inBase string, outBase string, dir string, conf *PathConfig, tree *PathConfigTree) {
inPath := inBase
if dir != "" {
inPath += "/" + dir
}
log.Noticef("reading input directory: %s", inPath)
files, err := ioutil.ReadDir(inPath)
2019-02-11 15:00:27 +01:00
if err != nil {
log.Panic(err)
}
2019-02-12 10:21:51 +01:00
tree.InputPath = inPath
2019-02-11 15:00:27 +01:00
// read config
var newConfig *PathConfig
log.Debug("looking for config.yml ...")
2019-02-12 10:21:51 +01:00
configFile := inPath + "/config.yml"
if _, err = os.Stat(configFile); os.IsNotExist(err) {
2019-02-11 15:00:27 +01:00
log.Debug("no config.yml found in this directory, using upper configs")
newConfig = conf
} else {
log.Debug("reading config...")
2019-02-12 10:21:51 +01:00
data, err := ioutil.ReadFile(configFile)
2019-02-11 15:00:27 +01:00
if err != nil {
2019-02-12 10:21:51 +01:00
log.Panicf("could not read file '%s': %s", configFile, err)
2019-02-11 15:00:27 +01:00
}
newConfig = new(PathConfig)
err = yaml.Unmarshal(data, newConfig)
if err != nil {
2019-02-12 10:21:51 +01:00
log.Panicf("could not parse YAML file '%s': %s", configFile, err)
2019-02-11 15:00:27 +01:00
}
log.Debug("merging config with upper config")
mergeConfig(newConfig, conf)
log.Debug(spew.Sdump(newConfig))
}
tree.Config = newConfig
2019-02-11 15:25:48 +01:00
// calc outDir
2019-02-12 10:21:51 +01:00
stripedDir := dir
2019-02-11 15:25:48 +01:00
regexStr := newConfig.Path.Strip
if regexStr != nil && *regexStr != "" {
2019-02-12 10:21:51 +01:00
if regex, err := regexp.Compile(*regexStr); err != nil {
log.Panicf("error compiling path.strip regex '%s' from '%s': %s", *regexStr, inBase+"/"+dir, err)
2019-02-11 15:25:48 +01:00
} else {
2019-02-12 10:21:51 +01:00
stripedDir = regex.ReplaceAllString(stripedDir, "")
2019-02-11 15:25:48 +01:00
}
}
2019-02-12 10:21:51 +01:00
outPath := outBase + "/" + stripedDir
outPath = strings.Replace(outPath, "//", "/", -1)
outPath = strings.TrimSuffix(outPath, "/")
2019-02-11 15:00:27 +01:00
2019-02-12 10:21:51 +01:00
log.Noticef("calculated output directory: %s", outPath)
tree.OutputPath = outPath
2019-02-11 15:00:27 +01:00
// first only files
for _, f := range files {
2019-02-12 10:21:51 +01:00
p := inPath + "/" + f.Name()
2019-02-11 15:00:27 +01:00
if strings.HasSuffix(f.Name(), ".md") {
log.Debugf(".MD %s", p)
2019-02-12 10:21:51 +01:00
if tree.InputFiles == nil {
tree.InputFiles = make([]string, 0)
}
tree.InputFiles = append(tree.InputFiles, f.Name())
2019-02-11 15:00:27 +01:00
}
}
// only directorys, needed config before
for _, f := range files {
2019-02-12 10:21:51 +01:00
p := inPath + "/" + f.Name()
2019-02-11 15:00:27 +01:00
if f.IsDir() {
log.Debugf("DIR %s", p)
newTree := new(PathConfigTree)
if tree.Sub == nil {
2019-02-12 10:21:51 +01:00
tree.Sub = make([]*PathConfigTree, 0)
2019-02-11 15:00:27 +01:00
}
tree.Sub = append(tree.Sub, newTree)
2019-02-12 10:21:51 +01:00
readContentDir(inPath, outPath, f.Name(), newConfig, newTree)
2019-02-11 15:00:27 +01:00
}
}
}
2019-02-11 15:25:48 +01:00
func processContent(conf *PathConfigTree) {
2019-02-12 10:21:51 +01:00
log.Debugf("trying to create output directory: %s", conf.OutputPath)
if dirH, err := os.Stat(conf.OutputPath); os.IsNotExist(err) {
err := os.MkdirAll(conf.OutputPath, 0755)
if err != nil {
log.Panicf("could not create output directory '%s': %s", conf.OutputPath, err)
}
log.Noticef("created output directory: %s", conf.OutputPath)
} else if dirH != nil {
if dirH.IsDir() {
log.Noticef("output directory '%s' already exists", conf.OutputPath)
} else {
log.Panicf("output directory '%s' is no directory", conf.OutputPath)
}
} else {
log.Panicf("unknown error for output directory '%s': %s", conf.OutputPath, err)
}
2019-02-11 15:25:48 +01:00
2019-02-12 10:21:51 +01:00
for _, el := range conf.Sub {
processContent(el)
}
2019-02-11 15:25:48 +01:00
}
2019-02-10 13:49:27 +01:00
func main() {
2019-02-11 15:00:27 +01:00
spew.Config.DisablePointerAddresses = true
spew.Config.DisableCapacities = true
spew.Config.DisableMethods = true
spew.Config.DisablePointerMethods = true
inDir = flag.String("in", "./", "input directory")
outDir = flag.String("out", "html", "output directory")
createOutDir := flag.Bool("create", false, "create output directory if not existing")
//clearOutDir := flag.Bool("clear", false, "clear output directory before generating website")
logLevel := flag.String("logLevel", "info", "log level: debug, info, warning, error")
flag.Parse()
logBackend := logging.NewLogBackend(os.Stderr, "", 0)
logBackendFormatter := logging.NewBackendFormatter(logBackend, logging.MustStringFormatter(
`%{color}%{time:15:04:05.000} %{shortfunc} ▶ %{level:.4s} %{id:03x}%{color:reset} %{message}`,
))
logBackendLeveled := logging.AddModuleLevel(logBackendFormatter)
logBackendLevel := logging.INFO
if logLevel != nil {
switch *logLevel {
case "debug":
logBackendLevel = logging.DEBUG
break
case "info":
logBackendLevel = logging.INFO
break
case "warning":
logBackendLevel = logging.WARNING
break
case "error":
logBackendLevel = logging.ERROR
break
}
}
logBackendLeveled.SetLevel(logBackendLevel, "")
logging.SetBackend(logBackendLeveled)
if inDir == nil || *inDir == "" {
log.Panic("input directory not specified")
}
log.Noticef("input directory: %s", *inDir)
if outDir == nil || *outDir == "" {
log.Panic("output directory not specified")
}
log.Noticef("output directory: %s", *outDir)
if createOutDir != nil && *createOutDir {
if _, err := os.Stat(*outDir); os.IsNotExist(err) {
log.Debugf("output directory '%s' does not exist", *outDir)
log.Debugf("trying to create output directory: %s", *outDir)
err := os.MkdirAll(*outDir, 0755)
if err != nil {
log.Panic(err)
}
log.Noticef("created output directory: %s", *outDir)
} else {
log.Noticef("output directory '%s' already exists", *outDir)
}
}
if fD, err := os.Stat(*outDir); os.IsNotExist(err) {
log.Panicf("output directory '%s' does not exist, try -create parameter or create manually", *outDir)
} else {
if fD == nil {
log.Panicf("something went wrong, could not get file handle for output dir %s", *outDir)
} else if !fD.IsDir() {
log.Panicf("output directory '%s' is not a directory", *outDir)
}
}
log.Debug("reading global config...")
p := *inDir + "/config.yml"
data, err := ioutil.ReadFile(p)
if err != nil {
log.Panicf("could not read file '%s': %s", p, err)
}
err = yaml.Unmarshal(data, globalConfig)
2019-02-10 13:49:27 +01:00
if err != nil {
2019-02-11 15:00:27 +01:00
log.Panicf("could not parse YAML file '%s': %s", p, err)
2019-02-10 13:49:27 +01:00
}
2019-02-11 15:00:27 +01:00
log.Debug(spew.Sdump(globalConfig))
log.Debugf("reading input directory %s", *inDir)
2019-02-12 10:21:51 +01:00
defaultInputFile := "README.md"
defaultOutputFile := "index.html"
defaultPathStrip := "^[0-9]*_"
defaultPathIgnoreForNav := "^_"
defaultFilenameStrip := ""
defaultFilenameIgnore := "_"
defaultFilenameOutputExtension := "html"
defaultPathConfig := new(PathConfig)
defaultPathConfig.InputFile = &defaultInputFile
defaultPathConfig.OutputFile = &defaultOutputFile
defaultPathConfig.Path.Strip = &defaultPathStrip
defaultPathConfig.Path.IgnoreForNav = &defaultPathIgnoreForNav
defaultPathConfig.Filename.Strip = &defaultFilenameStrip
defaultPathConfig.Filename.Ignore = &defaultFilenameIgnore
defaultPathConfig.Filename.OutputExtension = &defaultFilenameOutputExtension
readContentDir(*inDir+"/content", *outDir, "", defaultPathConfig, contentConfig)
//spew.Dump(contentConfig)
2019-02-11 15:00:27 +01:00
2019-02-11 15:25:48 +01:00
processContent(contentConfig)
2019-02-11 15:00:27 +01:00
/*
input, err := ioutil.ReadFile("README.md")
if err != nil {
panic(err)
}
//html := blackfriday.Run(input, blackfriday.WithRenderer(bfchroma.NewRenderer()))
html := blackfriday.Run(input)
fmt.Println(string(html))
*/
2019-02-10 13:49:27 +01:00
}