shiori/internal/cmd/utils.go

182 lines
3.8 KiB
Go
Raw Normal View History

2019-05-22 00:24:11 +08:00
package cmd
import (
"errors"
"fmt"
"io"
"net/http"
nurl "net/url"
"os"
2019-05-22 17:47:20 +08:00
"os/exec"
2019-05-22 00:24:11 +08:00
fp "path/filepath"
2019-05-22 17:47:20 +08:00
"runtime"
2019-05-22 00:24:11 +08:00
"strconv"
"strings"
"time"
"github.com/fatih/color"
"github.com/go-shiori/shiori/internal/model"
2019-05-22 17:47:20 +08:00
"golang.org/x/crypto/ssh/terminal"
2019-05-22 00:24:11 +08:00
)
var (
cIndex = color.New(color.FgHiCyan)
cSymbol = color.New(color.FgHiMagenta)
cTitle = color.New(color.FgHiGreen).Add(color.Bold)
cReadTime = color.New(color.FgHiMagenta)
cURL = color.New(color.FgHiYellow)
cExcerpt = color.New(color.FgHiWhite)
cTag = color.New(color.FgHiBlue)
2019-05-22 17:13:52 +08:00
cInfo = color.New(color.FgHiCyan)
cError = color.New(color.FgHiRed)
cWarning = color.New(color.FgHiYellow)
2019-05-22 00:24:11 +08:00
errInvalidIndex = errors.New("Index is not valid")
)
func normalizeSpace(str string) string {
2019-05-23 10:22:47 +08:00
str = strings.TrimSpace(str)
2019-05-22 00:24:11 +08:00
return strings.Join(strings.Fields(str), " ")
}
func isURLValid(s string) bool {
tmp, err := nurl.Parse(s)
return err == nil && tmp.Scheme != "" && tmp.Hostname() != ""
}
func clearUTMParams(url *nurl.URL) {
queries := url.Query()
for key := range queries {
if strings.HasPrefix(key, "utm_") {
queries.Del(key)
}
}
url.RawQuery = queries.Encode()
}
2019-05-24 14:25:29 +08:00
func downloadBookImage(url, dstPath string, timeout time.Duration) error {
2019-05-22 00:24:11 +08:00
// Fetch data from URL
client := &http.Client{Timeout: timeout}
resp, err := client.Get(url)
if err != nil {
return err
}
defer resp.Body.Close()
// Make sure destination directory exist
err = os.MkdirAll(fp.Dir(dstPath), os.ModePerm)
if err != nil {
return err
}
// Create destination file
dst, err := os.Create(dstPath)
if err != nil {
return err
}
defer dst.Close()
// Write response body to the file
_, err = io.Copy(dst, resp.Body)
if err != nil {
return err
}
return nil
}
func printBookmarks(bookmarks ...model.Bookmark) {
for _, bookmark := range bookmarks {
// Create bookmark index
strBookmarkIndex := fmt.Sprintf("%d. ", bookmark.ID)
strSpace := strings.Repeat(" ", len(strBookmarkIndex))
// Print bookmark title
cIndex.Print(strBookmarkIndex)
cTitle.Println(bookmark.Title)
// Print bookmark URL
cSymbol.Print(strSpace + "> ")
cURL.Println(bookmark.URL)
// Print bookmark excerpt
if bookmark.Excerpt != "" {
cSymbol.Print(strSpace + "+ ")
cExcerpt.Println(bookmark.Excerpt)
}
// Print bookmark tags
if len(bookmark.Tags) > 0 {
cSymbol.Print(strSpace + "# ")
for i, tag := range bookmark.Tags {
if i == len(bookmark.Tags)-1 {
cTag.Println(tag.Name)
} else {
cTag.Print(tag.Name + ", ")
}
}
}
// Append new line
fmt.Println()
}
}
// parseStrIndices converts a list of indices to their integer values
func parseStrIndices(indices []string) ([]int, error) {
var listIndex []int
for _, strIndex := range indices {
if !strings.Contains(strIndex, "-") {
index, err := strconv.Atoi(strIndex)
if err != nil || index < 1 {
return nil, errInvalidIndex
}
listIndex = append(listIndex, index)
continue
}
parts := strings.Split(strIndex, "-")
if len(parts) != 2 {
return nil, errInvalidIndex
}
minIndex, errMin := strconv.Atoi(parts[0])
maxIndex, errMax := strconv.Atoi(parts[1])
if errMin != nil || errMax != nil || minIndex < 1 || minIndex > maxIndex {
return nil, errInvalidIndex
}
for i := minIndex; i <= maxIndex; i++ {
listIndex = append(listIndex, i)
}
}
return listIndex, nil
}
2019-05-22 17:47:20 +08:00
// openBrowser tries to open the URL in a browser,
// and returns any error if it happened.
func openBrowser(url string) error {
var args []string
switch runtime.GOOS {
case "darwin":
args = []string{"open"}
case "windows":
args = []string{"cmd", "/c", "start"}
default:
args = []string{"xdg-open"}
}
cmd := exec.Command(args[0], append(args[1:], url)...)
return cmd.Run()
}
func getTerminalWidth() int {
width, _, _ := terminal.GetSize(int(os.Stdin.Fd()))
return width
}