golang 通过fsnotify监控文件,并通过文件变化重启程序

一、下载我们需要的包

> go get github.com/fsnotify/fsnotify

二、使用fsnotify监控文件

package main;

import (
	"github.com/fsnotify/fsnotify"
	"log"
	"fmt"
)

func main() {
	//创建一个监控对象
	watch, err := fsnotify.NewWatcher();
	if err != nil {
		log.Fatal(err);
	}
	defer watch.Close();
	//添加要监控的对象,文件或文件夹
	err = watch.Add("./tmp");
	if err != nil {
		log.Fatal(err);
	}
	//我们另启一个goroutine来处理监控对象的事件
	go func() {
		for {
			select {
			case ev := <-watch.Events:
				{
					//判断事件发生的类型,如下5种
					// Create 创建
					// Write 写入
					// Remove 删除
					// Rename 重命名
					// Chmod 修改权限
					if ev.Op&fsnotify.Create == fsnotify.Create {
						log.Println("创建文件 : ", ev.Name);
					}
					if ev.Op&fsnotify.Write == fsnotify.Write {
						log.Println("写入文件 : ", ev.Name);
					}
					if ev.Op&fsnotify.Remove == fsnotify.Remove {
						log.Println("删除文件 : ", ev.Name);
					}
					if ev.Op&fsnotify.Rename == fsnotify.Rename {
						log.Println("重命名文件 : ", ev.Name);
					}
					if ev.Op&fsnotify.Chmod == fsnotify.Chmod {
						log.Println("修改权限 : ", ev.Name);
					}
				}
			case err := <-watch.Errors:
				{
					log.Println("error : ", err);
					return;
				}
			}
		}
	}();

	//循环
	select {};
}

测试结果如下:

我们在tmp目录下的操作都被捕捉到了,但是fsnotify有一个问题,它无法递归的帮我们捕捉子目录、孙子目录的操作事件,这需要我们自已来实现。

还有一个问题就是当们修改文件夹名称时,fsnotify中event.Name仍然是原来的文件名,这就需要我们在重命名事件中,先移除之前的监控,然后添加新的监控。

修改如下:

package main;

import (
	"github.com/fsnotify/fsnotify"
	"fmt"
	"path/filepath"
	"os"
)

type Watch struct {
	watch *fsnotify.Watcher;
}

//监控目录
func (w *Watch) watchDir(dir string) {
	//通过Walk来遍历目录下的所有子目录
	filepath.Walk(dir, func(path string, info os.FileInfo, err error) error {
		//这里判断是否为目录,只需监控目录即可
		//目录下的文件也在监控范围内,不需要我们一个一个加
		if info.IsDir() {
			path, err := filepath.Abs(path);
			if err != nil {
				return err;
			}
			err = w.watch.Add(path);
			if err != nil {
				return err;
			}
			fmt.Println("监控 : ", path);
		}
		return nil;
	});
	go func() {
		for {
			select {
			case ev := <-w.watch.Events:
				{
					if ev.Op&fsnotify.Create == fsnotify.Create {
						fmt.Println("创建文件 : ", ev.Name);
						//这里获取新创建文件的信息,如果是目录,则加入监控中
						fi, err := os.Stat(ev.Name);
						if err == nil && fi.IsDir() {
							w.watch.Add(ev.Name);
							fmt.Println("添加监控 : ", ev.Name);
						}
					}
					if ev.Op&fsnotify.Write == fsnotify.Write {
						fmt.Println("写入文件 : ", ev.Name);
					}
					if ev.Op&fsnotify.Remove == fsnotify.Remove {
						fmt.Println("删除文件 : ", ev.Name);
						//如果删除文件是目录,则移除监控
						fi, err := os.Stat(ev.Name);
						if err == nil && fi.IsDir() {
							w.watch.Remove(ev.Name);
							fmt.Println("删除监控 : ", ev.Name);
						}
					}
					if ev.Op&fsnotify.Rename == fsnotify.Rename {
						fmt.Println("重命名文件 : ", ev.Name);
						//如果重命名文件是目录,则移除监控
						//注意这里无法使用os.Stat来判断是否是目录了
						//因为重命名后,go已经无法找到原文件来获取信息了
						//所以这里就简单粗爆的直接remove好了
						w.watch.Remove(ev.Name);
					}
					if ev.Op&fsnotify.Chmod == fsnotify.Chmod {
						fmt.Println("修改权限 : ", ev.Name);
					}
				}
			case err := <-w.watch.Errors:
				{
					fmt.Println("error : ", err);
					return;
				}
			}
		}
	}();
}

func main() {
	watch, _ := fsnotify.NewWatcher()
	w := Watch{
		watch: watch,
	}
	w.watchDir("./tmp");
	select {};
}

测试结果如下:

经过上面的例子,我们通过fsnotify来写一个监控配置文件,如果配置文件有修改,就重新启动服务。

我们先写一个可以运行的exe程序,server.go代码如下:

package main;

import (
	"io/ioutil"
	"log"
	"encoding/json"
	"net"
	"fmt"
	"os"
	"os/signal"
)

const (
	confFilePath = "./conf/conf.json";
)

//我们这里只是演示,配置项只设置一个
type Conf struct {
	Port int `json:port`;
}

func main() {
	//读取文件内容
	data, err := ioutil.ReadFile(confFilePath);
	if err != nil {
		log.Fatal(err);
	}
	var c Conf;
	//解析配置文件
	err = json.Unmarshal(data, &c);
	if err != nil {
		log.Fatal(err);
	}
	//根据配置项来监听端口
	lis, err := net.Listen("tcp", fmt.Sprintf(":%d", c.Port));
	if err != nil {
		log.Fatal(err);
	}
	log.Println("server start");
	go func() {
		ch := make(chan os.Signal);
		//获取程序退出信号
		signal.Notify(ch, os.Interrupt, os.Kill);
		<-ch;
		log.Println("server exit");
		os.Exit(1);
	}();
	for {
		conn, err := lis.Accept();
		if err != nil {
			continue;
		}
		go func(conn net.Conn) {
			defer conn.Close();
			conn.Write([]byte("hello\n"));
		}(conn);
	}
}

使用如下命令,编译成exe文件

> go build server.go

监控文件fsnotify3.go代码如下:

package main;

import (
	"github.com/fsnotify/fsnotify"
	"log"
	"fmt"
	"os/exec"
	"regexp"
	"strconv"
	"bytes"
	"errors"
	"os"
	"path/filepath"
)

const (
	confFilePath = "./conf";
)

//获取进程ID
func getPid(processName string) (int, error) {
	//通过wmic process get name,processid | findstr server.exe获取进程ID
	buf := bytes.Buffer{};
	cmd := exec.Command("wmic", "process", "get", "name,processid");
	cmd.Stdout = &buf;
	cmd.Run();
	cmd2 := exec.Command("findstr", processName);
	cmd2.Stdin = &buf;
	data, _ := cmd2.CombinedOutput();
	if len(data) == 0 {
		return -1, errors.New("not find");
	}
	info := string(data);
	//这里通过正则把进程id提取出来
	reg := regexp.MustCompile(`[0-9]+`);
	pid := reg.FindString(info);
	return strconv.Atoi(pid);
}

//启动进程
func startProcess(exePath string, args []string) error {
	attr := &os.ProcAttr{
		//files指定新进程继承的活动文件对象
		//前三个分别为,标准输入、标准输出、标准错误输出
		Files: []*os.File{os.Stdin, os.Stdout, os.Stderr},
		//新进程的环境变量
		Env: os.Environ(),
	}

	p, err := os.StartProcess(exePath, args, attr);
	if err != nil {
		return err;
	}
	fmt.Println(exePath, "进程启动");
	p.Wait();
	return nil;
}

func main() {
	//创建一个监控对象
	watch, err := fsnotify.NewWatcher();
	if err != nil {
		log.Fatal(err);
	}
	defer watch.Close();
	//添加要监控的文件
	err = watch.Add(confFilePath);
	if err != nil {
		log.Fatal(err);
	}
	//我们另启一个goroutine来处理监控对象的事件
	go func() {
		for {
			select {
			case ev := <-watch.Events:
				{
					//我们只需关心文件的修改
					if ev.Op&fsnotify.Write == fsnotify.Write {
						fmt.Println(ev.Name, "文件写入");
						//查找进程
						pid, err := getPid("server.exe");
						//获取运行文件的绝对路径
						exePath, _ := filepath.Abs("./server.exe")
						if err != nil {
							//启动进程
							go startProcess(exePath, []string{});
						} else {
							//找到进程,并退出
							process, err := os.FindProcess(pid);
							if err == nil {
								//让进程退出
								process.Kill();
								fmt.Println(exePath, "进程退出");
							}
							//启动进程
							go startProcess(exePath, []string{});
						}
					}
				}
			case err := <-watch.Errors:
				{
					fmt.Println("error : ", err);
					return;
				}
			}
		}
	}();

	//循环
	select {};
}

我们运行fsnotify3.go文件来监控我们的配置文件

通过上面的图可以看到,当我们修改配置文件中的端口号时,会先kill掉进程,然后再启动一个进程。

时间: 2024-07-29 18:36:50

golang 通过fsnotify监控文件,并通过文件变化重启程序的相关文章

解决往监控目录拖拽文件夹无法监控到的问题

在项目中一个应用场景是监控一个目录变化,但从监控目录外部拖拽一个文件夹进来,拖拽进来的文件夹里的文件无法监控到,通过查看资料,发现是一个参数的设置 #include <iostream> #include "DirectoryChanges.h" #include "MyDirectoryChangeHandler.h" int main() { wcout.imbue(locale("CHS")); CDirectoryChange

C# FileSystemWatcher 在监控文件夹和文件时的用法

概述 最近学习FileSystemWatcher的用法,它主要是监控一个文件夹,当文件夹内的文件要是有更改就要记录下来,我就整理下我对FileSystemWatcher 的理解和用法. FileSystemWatcher 用法 在应用FileSystemWatcher对象之前,你必须了解这个对象的一些基本属性和事件.毫无疑问,这个对象的最重要的属性为“EnableRaisingEvents”属性. 这个属性决定对象在收到改变通知时是否提交事件.如果EnableRaisingEvents属性设为假

Python 的 pyinotify 模块 监控文件夹和文件的变动

官方参考: https://github.com/seb-m/pyinotify/wiki/Events-types https://github.com/seb-m/pyinotify/wiki/Install 最近在网上看到python有个pyinotify模块,其中他们可以监控文件夹内的文件的创建,修改,读取,删除等系列操作,我修改了下,添加了可以吧操作记录写到日志里的一点方法,下面就贴出代码了给大家分享下:#!/usr/bin/env python ? 01 02 03 04 05 06

使用FileSystemWatcher监控文件夹及文件

引言 这一周主要精力集中学习一个同事开发的本地文件搜索项目上,其中客户端添加共享文件时主要是使用FileSystemWatcher 监控文件,并在各种事件发生时向服务器发送消息. 解决方法 FileSystemWatcher类以前也使用过,没有太仔细去观察,这次使用时才发现其事件提示会很有趣: a):当你添加文件或文件夹时,会触发Created事件,然后修改默认文件夹或文件名称再触发Changed事件. b):复制或移动文件夹文件时则是触发Created事件. c):删除文件夹或文件时触发Del

Ueditor1.4.3实现跨域上传到独立文件服务器,完美解决单文件和多文件上传!

再写配置方法之前先吐槽一下网上的各种教程,TM没一个有卵用,一群傻屌不会写就别写,写了就要负责. 百度google搜了半天,全是配置什么document.domain,根域名什么的,我只想对你说: 好了,talk is cheap show me the code,言归正传: 首先去ue官网下载1.43版 .net版本,解压之后,把demo文件夹改成web,把net那个文件夹拷贝出来放到web外面,如图: 这样就把编辑器的静态文件和动态文件分离了,动态部分放到文件服务器上,静态部分集成到我们的网

inotify-java linux系统监听文件发生变化,实时通知java程序

1 Overview     最近公司的一个任务需要实时监控文件系统中某个文件的内容变化.由于程序本身由Java编写,因此使用了inotify- java(http://code.google.com/p/inotify-java/).inotify-java只是对Linux中 inotify相关的内核调用进行了封装,因此在使用inotify-java之前有必要了解一下inotify.      inotify是一种基于inode的文件系统监控机制.从2.6.13-rc3版本起被集成到Linux

文件操作part3 文件内指针的移动

大前提:文件内指针移动是以Bytes为单位的,唯独在唯独在t模式下的read读取内容的个数是以字符为单位的 t模式下读取文件: with open('a.txt',mode='rt',encoding='utf-8') as f: data=f.read(3)#读取的是三个字符 print(data) with open('a.txt',mode='rb') as f: data=f.read(3)#读取的是三个Bytes print(data.decode('utf-8')) f.seek(指

脚本基础,文件查找,文件打包压缩。

本周知识点: ''' 1.文本处理工具和正则表达式 抽取文本的工具: 文件内容:cat,用于查看文件内容的基本命令,-E显示行结束符$ -n 显示行号 -A显示所有控制符-b 非空行编号 -s 压缩连续的空行成一行.hexdump :可以查看文件的ASCII值more :分页查看文件 -d 显示翻页及退出提示less : 一页一页地查看文件. /文本 搜索文本 n/N 跳到下一个或上一个匹配 显示文本前或后行内容 head : -c# 指定获取前#字节 -n# 指定获取前#行 (n可以省略)ta

微信文件传输助手文件夹在哪?一起来找找

微信文件传输助手是微信电脑版与手机微信之间相互传输图片等文件的好工具,但很多童鞋都找不到微信文件传输助手文件夹在哪,就让我们一起找找吧 1.先说说手机微信文件传输助手文件夹在哪吧 文件夹路径为/Tencent/MicroMsg/Download/ 2.电脑版微信文件传输助手文件夹在:/微信安装保存目录/wechat files/微信号/ 也可以点击接收到的图片下载保存到相应位置即可