LatexAutoInstaller/Main.go
2021-09-11 23:26:06 +02:00

274 lines
5.9 KiB
Go

package main
import (
"bufio"
"fmt"
"io"
"log"
"os"
"os/exec"
"path/filepath"
"regexp"
"runtime"
"strconv"
"strings"
)
const (
ErrNoCompiler = "none of the following latex compilers available: [latexmk, pdflatex]"
)
func main() {
fmt.Printf("Pdflatex command exists: %t\n", commandExists("pdflatex"))
fmt.Printf("LatexMk command exists: %t\n", commandExists("latexmk"))
fmt.Printf("Operation System: %s/%s\n", runtime.GOOS, runtime.GOARCH)
compileAndInstall()
}
func compileAndInstall() {
out, err := compileLatex()
if err != nil {
fmt.Println("An error occured while compiling the document!")
if err.Error() == ErrNoCompiler {
log.Fatal(err.Error())
}
if filename := parseMissingFile(out); filename != "" {
fmt.Printf("We need to download: %s\n", filename)
// now we need to perform a root check
if rootCheck() {
log.Println("Awesome! You are now running this program with root permissions!")
if installFile(filename) {
// we remove the main aux file to really trigger a rebuild!
os.Remove("main.aux")
// if successfully installed we try to compile again
compileAndInstall()
}
} else {
log.Fatal("This program must be run as root! (sudo)")
}
} else {
fmt.Println(*out)
fmt.Println("Another build error occured!")
}
} else {
fmt.Println("Document built successfully!")
}
}
func parseMissingFile(output *string) string {
matchfile := regexp.MustCompile("! LaTeX Error: File `([^`']*)' not found|! I can't find file `([^`']*)'.")
matches := matchfile.FindStringSubmatch(*output)
if matches != nil {
if matches[1] != "" {
return matches[1]
} else {
return matches[2]
}
}
// ok now we try to find a font error
fontregex := regexp.MustCompile(`! Font \\[^=]*=([^\s]*)\s`)
fontmatch := fontregex.FindStringSubmatch(*output)
if fontmatch != nil {
if fontmatch[1] != "" {
return fontmatch[1]
}
}
// now try babel errors
babelregex := regexp.MustCompile("Unknown option `([^`']*)'. Either you misspelled")
babelmatch := babelregex.FindStringSubmatch(*output)
if babelmatch != nil {
if babelmatch[1] != "" {
return babelmatch[1] + ".ldf"
}
}
return ""
}
// check if a specific system command is available
func commandExists(cmd string) bool {
_, err := exec.LookPath(cmd)
return err == nil
}
// compile latex document with passed cmd args (or default ones if nothing passed) and return output or error
func compileLatex() (*string, error) {
app := ""
if commandExists("latexmk") {
app = "latexmk"
} else if commandExists("pdflatex") {
app = "pdflatex"
} else {
return nil, fmt.Errorf(ErrNoCompiler)
}
argsWithoutProg := os.Args[1:]
filename := "main.tex"
if len(argsWithoutProg) > 0 {
filename = argsWithoutProg[len(argsWithoutProg)-1]
// cut last arg --> filename
argsWithoutProg = argsWithoutProg[:len(argsWithoutProg)-1]
}
addArgIfNotExisting(&argsWithoutProg, "-interaction", "-interaction=nonstopmode")
addArgIfNotExisting(&argsWithoutProg, "-synctex", "-synctex=1")
addArgIfNotExisting(&argsWithoutProg, "-ouput-format", "-output-format=pdf")
addArgIfNotExisting(&argsWithoutProg, "-file-line-error", "-file-line-error")
// insert default args
argsWithoutProg = append(argsWithoutProg, filename)
cmd := exec.Command(app, argsWithoutProg...)
stdout, _ := cmd.StdoutPipe()
stderr, _ := cmd.StderrPipe()
fmt.Println("Building:")
cmd.Start()
output := ""
go func() {
scanner := bufio.NewScanner(stdout)
for scanner.Scan() {
m := scanner.Text()
output += m + "\n"
printPoint()
}
fmt.Println()
}()
go func() {
scanner := bufio.NewScanner(stderr)
for scanner.Scan() {
scanner.Text()
printPoint()
}
}()
err := cmd.Wait()
return &output, err
}
func addArgIfNotExisting(args *[]string, searchstring string, arg string) {
exists := false
for _, s := range *args {
if strings.Contains(s, searchstring) {
exists = true
}
}
if !exists {
tmp := append(*args, arg)
*args = tmp
}
}
var i = 0
// printPoint print a point with every 10th method call
func printPoint() {
if i%10 == 0 {
fmt.Printf(".")
}
i++
if i > 500 {
i = 0
fmt.Println()
}
}
// rootCheck perform a root user check
func rootCheck() bool {
cmd := exec.Command("id", "-u")
output, err := cmd.Output()
if err != nil {
log.Fatal(err)
}
// output has trailing \n
// need to remove the \n
// otherwise it will cause error for strconv.Atoi
// log.Println(output[:len(output)-1])
// 0 = root, 501 = non-root user
i, err := strconv.Atoi(string(output[:len(output)-1]))
if err != nil {
// maybe no unix system?
log.Fatal(err)
}
return i == 0
}
func installFile(filename string) bool {
if commandExists("dnf") {
cmd := exec.Command("dnf", "-y", "install", fmt.Sprintf("tex(%s)", filename))
fmt.Println(cmd.String())
stdout, _ := cmd.StdoutPipe()
stderr, _ := cmd.StderrPipe()
fmt.Println("running dnf install now!")
cmd.Start()
printReadCloserToStdout(stdout)
printReadCloserToStdout(stderr)
err := cmd.Wait()
if err != nil {
fmt.Println(err.Error())
return false
}
return true
} else if commandExists("tlmgr") {
fmt.Println("dnf not existing -> trying to install with tlmgr")
// tlmgr package name should be filename without suffix
cmd := exec.Command("tlmgr", "install", strings.TrimSuffix(filename, filepath.Ext(filename)))
fmt.Println(cmd.String())
stdout, _ := cmd.StdoutPipe()
stderr, _ := cmd.StderrPipe()
fmt.Println("running tlmgr install now!")
cmd.Start()
printReadCloserToStdout(stdout)
printReadCloserToStdout(stderr)
err := cmd.Wait()
if err != nil {
fmt.Println(err.Error())
return false
}
return true
} else {
fmt.Println("There seems to be no tex distribution to be installed??")
return false
}
}
func printReadCloserToStdout(reader io.ReadCloser) {
go func() {
scanner := bufio.NewScanner(reader)
for scanner.Scan() {
m := scanner.Text()
fmt.Println(m)
}
}()
}