2019-05-22 00:24:11 +08:00
|
|
|
package cmd
|
|
|
|
|
|
|
|
import (
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"mime"
|
|
|
|
"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()
|
|
|
|
}
|
|
|
|
|
|
|
|
func downloadFile(url, dstPath string, timeout time.Duration) error {
|
|
|
|
// 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
|
|
|
|
}
|
|
|
|
|
|
|
|
// If destination path doesn't have extension, create it
|
|
|
|
if fp.Ext(dstPath) == "" {
|
|
|
|
cp := resp.Header.Get("Content-Type")
|
|
|
|
exts, err := mime.ExtensionsByType(cp)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("failed to create extension: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(exts) == 0 {
|
|
|
|
return fmt.Errorf("unknown content type")
|
|
|
|
}
|
|
|
|
|
|
|
|
dstPath += exts[0]
|
|
|
|
}
|
|
|
|
|
|
|
|
// 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
|
|
|
|
}
|