基于fsnotify的文件监控模块,为eSync自动编译加载提供更改监听功能!
Nevar pievienot vairāk kā 25 tēmas Tēmai ir jāsākas ar burtu vai ciparu, tā var saturēt domu zīmes ('-') un var būt līdz 35 simboliem gara.

313 rindas
7.8 KiB

pirms 4 gadiem
pirms 4 gadiem
pirms 4 gadiem
pirms 4 gadiem
pirms 4 gadiem
pirms 4 gadiem
pirms 4 gadiem
pirms 4 gadiem
pirms 4 gadiem
pirms 4 gadiem
pirms 4 gadiem
pirms 4 gadiem
pirms 4 gadiem
pirms 4 gadiem
pirms 4 gadiem
pirms 4 gadiem
pirms 4 gadiem
pirms 4 gadiem
pirms 4 gadiem
pirms 4 gadiem
pirms 4 gadiem
pirms 4 gadiem
pirms 4 gadiem
pirms 4 gadiem
pirms 4 gadiem
pirms 4 gadiem
pirms 4 gadiem
pirms 4 gadiem
pirms 4 gadiem
pirms 4 gadiem
pirms 4 gadiem
pirms 4 gadiem
pirms 4 gadiem
pirms 4 gadiem
pirms 4 gadiem
pirms 4 gadiem
pirms 4 gadiem
pirms 4 gadiem
pirms 4 gadiem
pirms 4 gadiem
pirms 4 gadiem
pirms 4 gadiem
pirms 4 gadiem
pirms 4 gadiem
pirms 4 gadiem
pirms 4 gadiem
pirms 4 gadiem
pirms 4 gadiem
pirms 4 gadiem
pirms 4 gadiem
pirms 4 gadiem
pirms 4 gadiem
pirms 4 gadiem
pirms 4 gadiem
  1. package main
  2. import (
  3. "bufio"
  4. "bytes"
  5. "encoding/binary"
  6. //"fmt"
  7. "github.com/fsnotify/fsnotify"
  8. "io"
  9. "net"
  10. "os"
  11. "path/filepath"
  12. "strings"
  13. "time"
  14. )
  15. const (
  16. SendDur = 2
  17. SleepDur = 86400
  18. )
  19. var CollectFiles map[string]struct{}
  20. var SendTimer *time.Timer
  21. var Str bytes.Buffer
  22. var Msg *bytes.Buffer
  23. var AddDirs []string
  24. var OnlyDirs []string
  25. var DelDirs []string
  26. var Conn net.Conn
  27. const (
  28. hrl = ".hrl"
  29. erl = ".erl"
  30. beam = ".beam"
  31. dtl = ".dtl"
  32. lfe = "lfe"
  33. ex = "ex"
  34. idea = ".idea"
  35. svn = ".svn"
  36. git = ".git"
  37. lock = ".lock"
  38. bea = ".bea"
  39. )
  40. type Watch struct {
  41. watch *fsnotify.Watcher
  42. }
  43. // 收集更改了的文件
  44. func CollectFile(File string) {
  45. ext := filepath.Ext(File)
  46. if ext != idea && ext != git && ext != svn && ext != lock && ext != bea && ext != "" && ext != idea && (ext == erl || ext == beam || ext == hrl || ext == ex || ext == dtl || ext == lfe) {
  47. CollectFiles[File] = struct{}{}
  48. SendTimer.Reset(time.Second * SendDur)
  49. }
  50. }
  51. // 发送文件列表到erl层
  52. func SendToErl() {
  53. // 拼写数据
  54. for k := range CollectFiles {
  55. Str.WriteString(k)
  56. Str.WriteString("\r\n")
  57. }
  58. CollectFiles = map[string]struct{}{}
  59. var length = int32(len(Str.Bytes()))
  60. //写入消息头
  61. _ = binary.Write(Msg, binary.BigEndian, length)
  62. //写入消息体
  63. _ = binary.Write(Msg, binary.BigEndian, Str.Bytes())
  64. Conn.Write(Msg.Bytes())
  65. Str.Reset()
  66. Msg.Reset()
  67. SendTimer.Reset(time.Second * SleepDur)
  68. }
  69. func isHidden(path string) bool {
  70. for i := len(path) - 1; i > 0; i-- {
  71. if path[i] != '.' {
  72. continue
  73. }
  74. if os.IsPathSeparator(path[i-1]) {
  75. return true
  76. }
  77. }
  78. if path[0] == '.' {
  79. return true
  80. }
  81. return false
  82. }
  83. func isOnlyDir(dirs []string, curDirs string) bool {
  84. cnt := 0
  85. for _, v := range dirs {
  86. if v != "" {
  87. cnt += 1
  88. if strings.Contains(curDirs[1:], v[1:]) {
  89. return true
  90. }
  91. }
  92. }
  93. if cnt == 0 {
  94. return true
  95. }
  96. return false
  97. }
  98. func isDelDir(dirs []string, curDirs string) bool {
  99. for _, v := range dirs {
  100. if v != "" {
  101. if strings.Contains(curDirs[1:], v[1:]) {
  102. return true
  103. }
  104. }
  105. }
  106. return false
  107. }
  108. //监控目录
  109. func (w *Watch) watchDir(dir string) {
  110. //通过Walk来遍历目录下的所有子目录
  111. filepath.Walk(dir, func(path string, info os.FileInfo, err error) error {
  112. //这里判断是否为目录,只需监控目录即可 目录下的文件也在监控范围内,不需要我们一个一个加
  113. if info.IsDir() {
  114. path, err := filepath.Abs(path)
  115. if err != nil {
  116. return err
  117. }
  118. if !isHidden(path) {
  119. if isOnlyDir(OnlyDirs, path) {
  120. if !isDelDir(DelDirs, path) {
  121. err = w.watch.Add(path)
  122. if err != nil {
  123. // fmt.Println("watch err : ", path, err)
  124. return err
  125. }
  126. // fmt.Println("watch success : ", path)
  127. }
  128. }
  129. }
  130. }
  131. return nil
  132. })
  133. for _, v := range AddDirs {
  134. if v != "" {
  135. //通过Walk来遍历目录下的所有子目录
  136. filepath.Walk(v, func(path string, info os.FileInfo, err error) error {
  137. //这里判断是否为目录,只需监控目录即可 目录下的文件也在监控范围内,不需要我们一个一个加
  138. if info.IsDir() {
  139. path, err := filepath.Abs(path)
  140. if err != nil {
  141. return err
  142. }
  143. if !isHidden(path) {
  144. err = w.watch.Add(path)
  145. if err != nil {
  146. // fmt.Println("watch AddDirs err : ", err)
  147. return err
  148. }
  149. // fmt.Println("watch AddDirs success : ", path)
  150. }
  151. }
  152. return nil
  153. })
  154. }
  155. }
  156. // 启动文件监听goroutine
  157. go func() {
  158. for {
  159. select {
  160. case ev := <-w.watch.Events:
  161. {
  162. if ev.Op&fsnotify.Create == fsnotify.Create {
  163. //这里获取新创建文件的信息,如果是目录,则加入监控中
  164. fi, err := os.Stat(ev.Name)
  165. if err == nil && fi.IsDir() {
  166. // 新建了文件夹
  167. // fmt.Println("创建文件夹 : ", ev.Name)
  168. if !isHidden(ev.Name) {
  169. if isOnlyDir(OnlyDirs, ev.Name) {
  170. if !isDelDir(DelDirs, ev.Name) {
  171. w.watch.Add(ev.Name)
  172. // fmt.Println("添加监控 : ", ev.Name)
  173. }
  174. }
  175. }
  176. } else {
  177. // 新建了文件
  178. CollectFile(ev.Name)
  179. // fmt.Println("创建文件 : ", ev.Name)
  180. }
  181. }
  182. if ev.Op&fsnotify.Write == fsnotify.Write {
  183. CollectFile(ev.Name)
  184. // fmt.Println("写入文件 : ", ev.Name)
  185. }
  186. if ev.Op&fsnotify.Remove == fsnotify.Remove {
  187. // fmt.Println("删除文件 : ", ev.Name)
  188. //如果删除文件是目录,则移除监控
  189. fi, err := os.Stat(ev.Name)
  190. if err == nil && fi.IsDir() {
  191. w.watch.Remove(ev.Name)
  192. // fmt.Println("删除监控 : ", ev.Name)
  193. }
  194. }
  195. if ev.Op&fsnotify.Rename == fsnotify.Rename {
  196. // fmt.Println("重命名文件 : ", ev.Name)
  197. //如果重命名文件是目录,则移除监控
  198. //注意这里无法使用os.Stat来判断是否是目录了
  199. //因为重命名后,go已经无法找到原文件来获取信息了
  200. //所以这里就简单粗爆的直接remove好了
  201. w.watch.Remove(ev.Name)
  202. }
  203. if ev.Op&fsnotify.Chmod == fsnotify.Chmod {
  204. // fmt.Println("修改权限 : ", ev.Name)
  205. }
  206. }
  207. case <-w.watch.Errors:
  208. {
  209. // fmt.Println("error : ", err)
  210. return
  211. }
  212. case <-SendTimer.C:
  213. SendToErl()
  214. }
  215. }
  216. }()
  217. }
  218. func read(reader *bufio.Reader) ([]byte, error) {
  219. // Peek 返回缓存的一个切片,该切片引用缓存中前 n 个字节的数据,
  220. // 该操作不会将数据读出,只是引用,引用的数据在下一次读取操作之
  221. // 前是有效的。如果切片长度小于 n,则返回一个错误信息说明原因。
  222. // 如果 n 大于缓存的总大小,则返回 ErrBufferFull。
  223. lengthByte, err := reader.Peek(4)
  224. if err != nil {
  225. return nil, err
  226. }
  227. //创建 Buffer缓冲器
  228. lengthBuff := bytes.NewBuffer(lengthByte)
  229. var length int32
  230. // 通过Read接口可以将buf中得内容填充到data参数表示的数据结构中
  231. err = binary.Read(lengthBuff, binary.BigEndian, &length)
  232. if err != nil {
  233. return nil, err
  234. }
  235. // Buffered 返回缓存中未读取的数据的长度
  236. if int32(reader.Buffered()) < length+4 {
  237. return nil, err
  238. }
  239. // 读取消息真正的内容
  240. pack := make([]byte, int(4+length))
  241. // Read 从 b 中读出数据到 p 中,返回读出的字节数和遇到的错误。
  242. // 如果缓存不为空,则只能读出缓存中的数据,不会从底层 io.Reader
  243. // 中提取数据,如果缓存为空,则:
  244. // 1、len(p) >= 缓存大小,则跳过缓存,直接从底层 io.Reader 中读
  245. // 出到 p 中。
  246. // 2、len(p) < 缓存大小,则先将数据从底层 io.Reader 中读取到缓存
  247. // 中,再从缓存读取到 p 中。
  248. _, err = reader.Read(pack)
  249. if err != nil {
  250. return nil, err
  251. }
  252. return pack[4:], nil
  253. }
  254. func main() {
  255. CollectFiles = map[string]struct{}{}
  256. SendTimer = time.NewTimer(time.Second * SleepDur)
  257. defer SendTimer.Stop()
  258. Addr := "localhost:" + os.Args[2]
  259. var err error
  260. Conn, err = net.Dial("tcp", Addr)
  261. if err != nil {
  262. //fmt.Println("IMY****************建立tcp失败 : ", Addr)
  263. return
  264. }
  265. // 建立tcp 连接后需要从erlSync 接受监听目录相关配置
  266. var reader *bufio.Reader
  267. reader = bufio.NewReader(Conn)
  268. data, err := read(reader)
  269. if err == io.EOF {
  270. //fmt.Println("IMY****************Tcp 断开链接 : ", Addr)
  271. return
  272. }
  273. if err != nil {
  274. //fmt.Println("IMY****************Tcp read err : ", err)
  275. return
  276. }
  277. //fmt.Println("IMY****************建立tcp成功 : ", os.Args[0])
  278. //fmt.Println("IMY****************Tcp read data : ", string(data))
  279. dirs := strings.Split(string(data), "\r\n")
  280. AddDirs = strings.Split(dirs[0], "|")
  281. OnlyDirs = strings.Split(dirs[1], "|")
  282. DelDirs = strings.Split(dirs[2], "|")
  283. Msg = new(bytes.Buffer)
  284. watch, _ := fsnotify.NewWatcher()
  285. w := Watch{watch: watch}
  286. w.watchDir(os.Args[1])
  287. data = make([]byte, 10)
  288. _, _ = Conn.Read(data)
  289. Conn.Close()
  290. // fmt.Println("IMY****************tcp关闭了 : ", os.Args[0])
  291. }