-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathfile.go
157 lines (134 loc) · 3.37 KB
/
file.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
// Copyright 2021 Wayback Archiver. All rights reserved.
// Use of this source code is governed by the MIT
// license that can be found in the LICENSE file.
/*
Package helper handles common functions for the waybackk application in Golang.
*/
package helper // import "github.com/wabarc/helper"
import (
"bufio"
"fmt"
"io"
"mime"
"net/url"
"os"
"os/exec"
"strings"
"time"
"github.com/kennygrant/sanitize"
)
// FileName returns filename from webpage's link and content type.
func FileName(link, contentType string) string {
now := time.Now().Format("2006-01-02-150405.000")
ext := ".html"
if exts, _ := mime.ExtensionsByType(contentType); len(exts) > 0 {
ext = exts[0]
if strings.HasSuffix(ext, "htm") {
ext = strings.ReplaceAll(ext, "htm", "html")
}
}
for _, e := range []string{"jpe", "jpeg"} {
if strings.HasSuffix(ext, e) {
ext = strings.ReplaceAll(ext, e, "jpg")
}
}
u, err := url.ParseRequestURI(link)
if err != nil || u.Scheme == "" || u.Hostname() == "" {
return now + ext
}
domain := strings.ReplaceAll(u.Hostname(), ".", "-")
if u.Path == "" || u.Path == "/" {
return fmt.Sprintf("%s-%s%s", now, domain, ext)
}
baseName := strings.TrimPrefix(sanitize.BaseName(u.Path), "-")
if parts := strings.Split(baseName, "-"); len(parts) > 4 {
baseName = strings.Join(parts[:4], "-")
}
return fmt.Sprintf("%s-%s-%s%s", now, domain, baseName, ext)
}
// FileSeze returns file attritubes of size about an inode, and
// it's unit alway is bytes.
func FileSize(filepath string) int64 {
f, err := os.Stat(filepath)
if err != nil {
return 0
}
return f.Size()
}
// Exists reports whether the named file or directory exists.
func Exists(name string) bool {
if _, err := os.Stat(name); err != nil {
if os.IsNotExist(err) {
return false
}
}
return true
}
// MoveFile move file to another directory.
func MoveFile(src, dst string) error {
if src == dst {
return nil
}
in, err := os.Open(src)
if err != nil {
return fmt.Errorf("Couldn't open source file: %s", err)
}
si, err := in.Stat()
if err != nil {
return fmt.Errorf("Stat error: %s", err)
}
flag := os.O_WRONLY | os.O_CREATE | os.O_TRUNC
perm := si.Mode() & os.ModePerm
out, err := os.OpenFile(dst, flag, perm)
if err != nil {
return err
}
defer out.Close()
_, err = io.Copy(out, in)
in.Close()
if err != nil {
return fmt.Errorf("Writing to output file failed: %s", err)
}
err = out.Sync()
if err != nil {
return fmt.Errorf("Sync error: %s", err)
}
err = os.Remove(src)
if err != nil {
return fmt.Errorf("Failed removing original file: %s", err)
}
return nil
}
// WriteFile writes byte slices to a specified path; it will be created
// if it does not exist. It returns an error.
func WriteFile(path string, data []byte, mode os.FileMode) error {
if data == nil {
return fmt.Errorf("no data write to: %s", path)
}
file, err := os.OpenFile(path, os.O_WRONLY|os.O_CREATE, mode)
if err != nil {
return err
}
defer file.Close()
writer := bufio.NewWriter(file)
if _, err = writer.Write(data); err != nil {
return err
}
return writer.Flush()
}
// WebPToPNG convert WebP to PNG
func WebPToPNG(src, dst string) error {
dwebp, err := exec.LookPath("dwebp")
if err != nil {
return err
}
args := []string{src, "-o", dst}
cmd := exec.Command(dwebp, args...)
if err := cmd.Start(); err != nil {
return err
}
if err := cmd.Wait(); err != nil {
return err
}
return nil
}