glow/main.go

162 lines
3.2 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"
"path/filepath"
2019-11-04 23:17:36 +00:00
"github.com/mattn/go-isatty"
"github.com/spf13/cobra"
2019-12-19 00:01:26 +00:00
"github.com/charmbracelet/glamour"
"github.com/charmbracelet/glamour/ansi"
2019-11-04 23:17:36 +00:00
)
var (
Version = ""
CommitSHA = ""
readmeNames = []string{"README.md", "README"}
style string
width uint
rootCmd = &cobra.Command{
2019-12-20 21:47:47 +00:00
Use: "glow SOURCE",
Short: "Render markdown on the CLI, with pizzazz!",
SilenceErrors: false,
SilenceUsage: false,
2019-11-24 01:40:29 +00:00
RunE: execute,
}
)
type Source struct {
reader io.ReadCloser
URL string
}
func readerFromArg(s string) (*Source, error) {
if s == "-" {
return &Source{reader: os.Stdin}, nil
}
2019-12-10 16:18:59 +00:00
// a GitHub or GitLab URL (even without the protocol):
2019-11-25 03:55:09 +00:00
if u, ok := isGitHubURL(s); 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
}
2019-11-25 05:55:50 +00:00
if u, ok := isGitLabURL(s); 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:
if u, err := url.ParseRequestURI(s); 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)
}
return &Source{resp.Body, u.String()}, nil
}
}
2019-12-10 16:18:59 +00:00
// a valid file or directory:
st, err := os.Stat(s)
if len(s) == 0 || (err == nil && st.IsDir()) {
for _, v := range readmeNames {
2019-12-10 14:55:49 +00:00
n := filepath.Join(s, v)
r, err := os.Open(n)
if err == nil {
2019-12-10 14:55:49 +00:00
u, _ := filepath.Abs(n)
2019-11-25 23:34:15 +00:00
return &Source{r, u}, nil
}
}
return nil, errors.New("missing markdown source")
}
r, err := os.Open(s)
2019-11-25 23:34:15 +00:00
u, _ := filepath.Abs(s)
return &Source{r, u}, err
}
2019-11-24 01:40:29 +00:00
func execute(cmd *cobra.Command, args []string) error {
var arg string
if len(args) > 0 {
arg = args[0]
2019-11-09 22:12:54 +00:00
}
2019-12-10 16:18:59 +00:00
// create an io.Reader from the markdown source in cli-args
src, err := readerFromArg(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
}
2019-12-10 16:18:59 +00:00
// 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 !isatty.IsTerminal(os.Stdout.Fd()) &&
!cmd.Flags().Changed("style") {
style = "notty"
}
u, err := url.ParseRequestURI(src.URL)
if err == nil {
u.Path = filepath.Dir(u.Path)
}
2019-12-19 00:01:26 +00:00
r, err := glamour.NewTermRenderer(style, ansi.Options{
BaseURL: u.String() + "/",
WordWrap: int(width),
})
if err != nil {
return err
}
out, err := r.RenderBytes(b)
fmt.Printf("%s", string(out))
return err
}
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
2019-11-26 20:03:46 +00:00
rootCmd.Flags().StringVarP(&style, "style", "s", "dark", "style name or JSON path")
rootCmd.Flags().UintVarP(&width, "width", "w", 100, "word-wrap at width")
2019-11-04 23:17:36 +00:00
}