|
|
- package main
-
- import (
- "bytes"
- "encoding/binary"
- "io"
- "io/fs"
- "os"
- "path/filepath"
- "strings"
- "time"
-
- "github.com/fsnotify/fsnotify"
- )
-
- const (
- SendDur = 1111 // 发送时间间隔毫秒
- SleepDur = 86400000 // 定期器初始睡眠时间
- )
-
- var CollectFiles map[string]struct{}
- var CollectExt map[string]struct{}
- var SendTimer *time.Timer
- var Str bytes.Buffer
- var AddExtraDirs []string
- var AddOnlyDirs []string
- var OnlyDirs []string
- var DelDirs []string
- var LenBuff []byte
-
- type Watch struct {
- watch *fsnotify.Watcher
- }
-
- const (
- //LOGPATH LOGPATH/time.Now().Format(FORMAT)/*.log
- LOGPATH = "./"
- //FORMAT .
- FORMAT = "20060102"
- //LineFeed 换行
- LineFeed = "\r\n"
- FileName = "fileSync.log"
- )
-
- //以天为基准,存日志
- var path = LOGPATH + time.Now().Format(FORMAT) + "_"
-
- //WriteLog return error
- func WriteLog(msg string) error {
- var (
- err error
- f *os.File
- )
-
- f, err = os.OpenFile(path+FileName, os.O_CREATE|os.O_WRONLY|os.O_APPEND, 0666)
- _, err = io.WriteString(f, "Tag"+"::"+msg+LineFeed)
-
- defer f.Close()
- return err
- }
-
- //IsExist 判断文件夹/文件是否存在 存在返回 true
- func IsExist(f string) bool {
- _, err := os.Stat(f)
- return err == nil || os.IsExist(err)
- }
-
- // 收集更改了的文件
- func CollectFile(File string) {
- ext := filepath.Ext(File)
- if _, ok := CollectExt[ext]; ok{
- CollectFiles[File] = struct{}{}
- SendTimer.Reset(time.Millisecond * SendDur)
- }
- }
-
- func isHidden(path string) bool {
- for i := len(path) - 1; i > 0; i-- {
- if path[i] != '.' {
- continue
- }
- if os.IsPathSeparator(path[i-1]) {
- return true
- }
- }
- if path[0] == '.' {
- return true
- }
- return false
- }
-
- func isOnlyDir(dirs []string, curDirs string) bool {
- cnt := 0
- for _, v := range dirs {
- if v != "" {
- cnt += 1
- if strings.Contains(curDirs[1:], v[1:]) {
- return true
- }
- }
- }
- if cnt == 0 {
- return true
- }
- return false
- }
-
- func isDelDir(dirs []string, curDirs string) bool {
- for _, v := range dirs {
- if v != "" {
- if strings.Contains(curDirs[1:], v[1:]) {
- return true
- }
- }
- }
-
- return false
- }
-
- // 判断所给路径文件/文件夹是否存在
- func existPath(path string) bool {
- _, err := os.Stat(path) //os.Stat获取文件信息
- if err != nil {
- if os.IsNotExist(err) {
- return false
- }
- return true
- }
- return true
- }
-
- //监控目录
- func (w *Watch) watchDir(dir string) {
- // 启动文件监听goroutine
- go func() {
- for {
- select {
- case ev := <-w.watch.Events:
- {
- if ev.Has(fsnotify.Create) {
- //这里获取新创建文件的信息,如果是目录,则加入监控中
- fi, err := os.Stat(ev.Name)
- if err == nil && fi.IsDir() {
- // 新建了文件夹
- if !isHidden(ev.Name) {
- if isOnlyDir(OnlyDirs, ev.Name) {
- if !isDelDir(DelDirs, ev.Name) {
- w.watch.Add(ev.Name)
- }
- }
- }
- } else {
- // 新建了文件
- CollectFile(ev.Name)
- }
- }
- if ev.Has(fsnotify.Write) {
- CollectFile(ev.Name)
- }
- if ev.Has(fsnotify.Remove) {
- //如果删除文件是目录,则移除监控
- fi, err := os.Stat(ev.Name)
- if err == nil && fi.IsDir() {
- w.watch.Remove(ev.Name)
- }
- }
- if ev.Has(fsnotify.Rename) {
- //如果重命名文件是目录,则移除监控
- //注意这里无法使用os.Stat来判断是否是目录了
- //因为重命名后,go已经无法找到原文件来获取信息了
- //所以这里就简单粗爆的直接remove好了
- w.watch.Remove(ev.Name)
- }
- if ev.Has(fsnotify.Chmod) {
- }
- }
- case <-w.watch.Errors:
- {
- continue
- }
- case <-SendTimer.C:
- SendToErl()
- }
- }
- }()
-
- //通过Walk来遍历目录下的所有子目录
- filepath.WalkDir(dir, func(path string, info fs.DirEntry, err error) error {
- //这里判断是否为目录,只需监控目录即可 目录下的文件也在监控范围内,不需要我们一个一个加
-
- if err == nil && info.IsDir() {
- path, err := filepath.Abs(path)
- if err != nil {
- return err
- }
- if !isHidden(path) && isOnlyDir(OnlyDirs, path) && !isDelDir(DelDirs, path) {
- err = w.watch.Add(path)
- if err != nil {
- return err
- }
- }
- }
- return nil
- })
-
- for _, v := range AddExtraDirs {
- if v != "" && existPath(v) {
- //通过Walk来遍历目录下的所有子目录
- filepath.WalkDir(v, func(path string, info fs.DirEntry, err error) error {
- //这里判断是否为目录,只需监控目录即可 目录下的文件也在监控范围内,不需要我们一个一个加
- if err == nil && info.IsDir() {
- path, err := filepath.Abs(path)
- if err != nil {
- return err
- }
- if !isHidden(path) {
- err = w.watch.Add(path)
- if err != nil {
- return err
- }
- }
- }
- return nil
- })
- }
- }
-
- for _, v := range AddOnlyDirs {
- if v != "" && existPath(v) {
- w.watch.Add(v)
- }
- }
- }
-
- //********************************************** port start ************************************************************
- func ReadLen() (int32, error) {
- if _, err := io.ReadFull(os.Stdin, LenBuff); err != nil {
- return 0, err
- }
- size := int32(binary.BigEndian.Uint32(LenBuff))
-
- return size, nil
- }
-
- func Read() ([]byte, error) {
- len, err := ReadLen()
- if err != nil {
- return nil, err
- } else if len == 0 {
- return []byte{}, nil
- }
- data := make([]byte, len)
- size, err := io.ReadFull(os.Stdin, data)
- return data[:size], err
- }
-
- func Write(data []byte) (int, error) {
- size := len(data)
- binary.BigEndian.PutUint32(LenBuff, uint32(size))
-
- if _, err := os.Stdout.Write(LenBuff); err != nil {
- return 0, err
- }
-
- return os.Stdout.Write(data)
- }
-
- // 发送文件列表到erl层
- func SendToErl() {
- for k := range CollectFiles {
- Str.WriteString(k)
- Str.WriteString("\r\n")
- }
- CollectFiles = map[string]struct{}{}
- Write(Str.Bytes())
- Str.Reset()
- }
-
- //********************************************** port end ************************************************************
-
- func main() {
- CollectFiles = map[string]struct{}{}
- CollectExt = map[string]struct{}{}
- LenBuff = make([]byte, 4)
-
- SendTimer = time.NewTimer(time.Millisecond * SleepDur)
- defer SendTimer.Stop()
-
- Write([]byte("init"))
-
- args := os.Args
-
- ExtList := strings.Split(args[2], "|")
- for _, v := range ExtList {
- CollectExt[v] = struct{}{}
- }
-
- dirs := strings.Split(args[3], "\r\n")
- AddExtraDirs = strings.Split(dirs[0], "|")
- AddOnlyDirs = strings.Split(dirs[1], "|")
- OnlyDirs = strings.Split(dirs[2], "|")
- DelDirs = strings.Split(dirs[3], "|")
-
- watch, _ := fsnotify.NewWatcher()
- defer watch.Close()
- w := Watch{watch: watch}
- w.watchDir(args[1])
- Read()
- }
|