glow/main.go

306 lines
6.4 KiB
Go
Raw Normal View History

2019-11-04 23:17:36 +00:00
package main
import (
"errors"
2019-11-04 23:17:36 +00:00
"fmt"
"io"
2019-11-09 22:12:54 +00:00
"io/ioutil"
"net/http"
"net/url"
2019-11-09 20:07:01 +00:00
"os"
"os/exec"
"path/filepath"
"strings"
2019-11-04 23:17:36 +00:00
2020-06-20 23:54:01 +00:00
"github.com/meowgorithm/babyenv"
"github.com/spf13/cobra"
"golang.org/x/crypto/ssh/terminal"
2020-05-26 20:28:53 +00:00
tea "github.com/charmbracelet/bubbletea"
2019-12-19 00:01:26 +00:00
"github.com/charmbracelet/glamour"
"github.com/charmbracelet/glow/ui"
2019-11-04 23:17:36 +00:00
)
var (
Version = ""
CommitSHA = ""
readmeNames = []string{"README.md", "README"}
pager bool
style string
width uint
rootCmd = &cobra.Command{
Use: "glow SOURCE",
Short: "Render markdown on the CLI, with pizzazz!",
SilenceErrors: false,
SilenceUsage: false,
TraverseChildren: true,
RunE: execute,
}
)
2020-03-31 06:57:49 +00:00
// source provides a readable markdown source.
2020-03-31 06:53:35 +00:00
type source struct {
reader io.ReadCloser
URL string
}
2020-03-31 06:57:49 +00:00
// sourceFromArg parses an argument and creates a readable source for it.
func sourceFromArg(arg string) (*source, error) {
2020-03-31 06:53:35 +00:00
// from stdin
2020-03-31 06:57:49 +00:00
if arg == "-" {
2020-03-31 06:53:35 +00:00
return &source{reader: os.Stdin}, nil
}
2019-12-10 16:18:59 +00:00
// a GitHub or GitLab URL (even without the protocol):
2020-03-31 06:57:49 +00:00
if u, ok := isGitHubURL(arg); ok {
src, err := findGitHubREADME(u)
2019-11-25 03:55:09 +00:00
if err != nil {
return nil, err
}
return src, nil
2019-11-25 03:55:09 +00:00
}
2020-03-31 06:57:49 +00:00
if u, ok := isGitLabURL(arg); ok {
src, err := findGitLabREADME(u)
2019-11-25 05:55:50 +00:00
if err != nil {
return nil, err
}
return src, nil
2019-11-25 05:55:50 +00:00
}
2019-11-25 03:55:09 +00:00
2019-12-10 16:18:59 +00:00
// HTTP(S) URLs:
2020-03-31 06:57:49 +00:00
if u, err := url.ParseRequestURI(arg); err == nil {
2019-11-25 23:34:15 +00:00
if u.Scheme != "" {
if u.Scheme != "http" && u.Scheme != "https" {
return nil, fmt.Errorf("%s is not a supported protocol", u.Scheme)
}
2019-11-25 23:34:15 +00:00
resp, err := http.Get(u.String())
if err != nil {
return nil, err
}
if resp.StatusCode != http.StatusOK {
return nil, fmt.Errorf("HTTP status %d", resp.StatusCode)
}
2020-03-31 06:53:35 +00:00
return &source{resp.Body, u.String()}, nil
}
}
// a directory:
2020-03-31 06:57:49 +00:00
if len(arg) == 0 {
// use the current working dir if no argument was supplied
2020-03-31 06:57:49 +00:00
arg = "."
}
2020-03-31 06:57:49 +00:00
st, err := os.Stat(arg)
if err == nil && st.IsDir() {
2020-03-31 06:53:35 +00:00
var src *source
2020-03-31 06:57:49 +00:00
_ = filepath.Walk(arg, func(path string, info os.FileInfo, err error) error {
for _, v := range readmeNames {
if strings.EqualFold(filepath.Base(path), v) {
r, err := os.Open(path)
if err != nil {
continue
}
u, _ := filepath.Abs(path)
2020-03-31 06:53:35 +00:00
src = &source{r, u}
// abort filepath.Walk
return errors.New("source found")
}
}
return nil
})
2020-03-31 06:53:35 +00:00
if src != nil {
return src, nil
}
return nil, errors.New("missing markdown source")
}
// a file:
2020-03-31 06:57:49 +00:00
r, err := os.Open(arg)
u, _ := filepath.Abs(arg)
2020-03-31 06:53:35 +00:00
return &source{r, u}, err
}
2020-05-05 05:48:05 +00:00
func validateOptions(cmd *cobra.Command) {
isTerminal := terminal.IsTerminal(int(os.Stdout.Fd()))
// We want to use a special no-TTY style, when stdout is not a terminal
// and there was no specific style passed by arg
if !isTerminal && !cmd.Flags().Changed("style") {
style = "notty"
}
// Detect terminal width
if isTerminal && !cmd.Flags().Changed("width") {
w, _, err := terminal.GetSize(int(os.Stdout.Fd()))
if err == nil {
width = uint(w)
}
}
if width == 0 {
width = 80
}
if width > 120 {
width = 120
}
}
2019-11-24 01:40:29 +00:00
func execute(cmd *cobra.Command, args []string) error {
2020-05-05 05:48:05 +00:00
validateOptions(cmd)
2020-01-08 16:23:04 +00:00
if len(args) == 0 {
return executeArg(cmd, "", os.Stdout)
}
for _, arg := range args {
if err := executeArg(cmd, arg, os.Stdout); err != nil {
return err
}
}
return nil
}
func executeArg(cmd *cobra.Command, arg string, w io.Writer) error {
2020-05-13 23:15:39 +00:00
2020-05-25 15:45:56 +00:00
// Only run TUI if there are no arguments (excluding flags)
if arg == "" {
2020-06-20 23:54:01 +00:00
// Read environment to get debugging stuff
2020-08-07 16:34:48 +00:00
var cfg ui.Config
2020-06-20 23:54:01 +00:00
if err := babyenv.Parse(&cfg); err != nil {
return fmt.Errorf("error parsing config: %v", err)
}
2020-08-07 16:34:48 +00:00
// Config from flags
2020-08-24 12:53:55 +00:00
// cfg.IdentityFile = identityFile
2020-08-07 16:34:48 +00:00
2020-06-20 23:54:01 +00:00
// Log to file, if set
if cfg.Logfile != "" {
f, err := tea.LogToFile(cfg.Logfile, "glow")
if err != nil {
return err
}
defer f.Close()
}
2020-05-13 23:15:39 +00:00
2020-06-20 23:54:01 +00:00
// Run Bubble Tea program
p := ui.NewProgram(style, cfg)
p.EnterAltScreen()
if err := p.Start(); err != nil {
return err
}
p.ExitAltScreen()
2020-06-23 16:03:00 +00:00
p.DisableMouseCellMotion()
2020-05-13 23:15:39 +00:00
2020-06-20 23:54:01 +00:00
// Exit message
2020-05-13 23:22:46 +00:00
fmt.Printf("\n Thanks for using Glow!\n\n")
2020-05-13 23:15:39 +00:00
return nil
2020-01-08 16:23:04 +00:00
}
2019-12-10 16:18:59 +00:00
// create an io.Reader from the markdown source in cli-args
2020-03-31 06:57:49 +00:00
src, err := sourceFromArg(arg)
if err != nil {
return err
2019-11-09 22:12:54 +00:00
}
defer src.reader.Close()
b, err := ioutil.ReadAll(src.reader)
if err != nil {
return err
}
// render
2019-12-23 03:27:27 +00:00
var baseURL string
u, err := url.ParseRequestURI(src.URL)
if err == nil {
u.Path = filepath.Dir(u.Path)
2019-12-23 03:27:27 +00:00
baseURL = u.String() + "/"
}
2020-03-31 06:53:35 +00:00
// initialize glamour
var gs glamour.TermRendererOption
if style == "auto" {
gs = glamour.WithEnvironmentConfig()
} else {
gs = glamour.WithStylePath(style)
}
r, err := glamour.NewTermRenderer(
gs,
glamour.WithWordWrap(int(width)),
glamour.WithBaseURL(baseURL),
)
if err != nil {
return err
}
out, err := r.RenderBytes(b)
if err != nil {
return err
}
// trim lines
lines := strings.Split(string(out), "\n")
var content string
for i, s := range lines {
content += strings.TrimSpace(s)
2019-12-31 05:36:15 +00:00
// don't add an artificial newline after the last split
if i+1 < len(lines) {
content += "\n"
}
}
// display
if cmd.Flags().Changed("pager") {
pager := os.Getenv("PAGER")
if pager == "" {
pager = "less -r"
}
pa := strings.Split(pager, " ")
c := exec.Command(pa[0], pa[1:]...)
c.Stdin = strings.NewReader(content)
c.Stdout = os.Stdout
return c.Run()
}
fmt.Fprint(w, content)
return nil
}
func main() {
if err := rootCmd.Execute(); err != nil {
os.Exit(-1)
}
}
func init() {
2019-12-09 13:01:47 +00:00
if len(CommitSHA) >= 7 {
vt := rootCmd.VersionTemplate()
2019-12-09 13:01:47 +00:00
rootCmd.SetVersionTemplate(vt[:len(vt)-1] + " (" + CommitSHA[0:7] + ")\n")
}
if Version == "" {
Version = "unknown (built from source)"
}
rootCmd.Version = Version
2020-05-25 15:45:56 +00:00
// "Glow Classic" cli arguments
rootCmd.Flags().BoolVarP(&pager, "pager", "p", false, "display with pager")
2020-01-14 01:20:21 +00:00
rootCmd.Flags().StringVarP(&style, "style", "s", "auto", "style name or JSON path")
rootCmd.Flags().UintVarP(&width, "width", "w", 0, "word-wrap at width")
2020-04-23 22:05:38 +00:00
2020-05-25 15:45:56 +00:00
// For network-related operations, namely stashing and the TUI
2020-08-24 12:53:55 +00:00
/*
rootCmd.PersistentFlags().StringVarP(&identityFile, "identity", "i", "", "path to identity file (that is, an ssh private key)")
rootCmd.PersistentFlags().BoolVarP(&forceKey, "force-key", "f", false, "force the use of the SSH key on disk (that is, ignore ssh-agent)")
*/
2020-05-25 15:45:56 +00:00
// Stash
2020-04-23 22:05:38 +00:00
stashCmd.PersistentFlags().StringVarP(&memo, "memo", "m", "", "memo/note for stashing")
rootCmd.AddCommand(stashCmd)
2019-11-04 23:17:36 +00:00
}