我们平时在集群环境的时候加锁常常使用redis,如果就单pod实例某个操作想加锁再操作的话,不是用redis,可以考虑使用文件锁。

通常我们需要把nodejs的api文档放一下,这样方便查找一些api

File system | Node.js v14.18.0 Documentation

一、File文件加锁

1、open版本

一开始通过百度回发现使用fs的open方法进行加锁。

var fs = require("fs");

fs.open("test.txt","wx",function(err,fp){
 if(err) return console.error(err);
 // 这里面可以进行我们的操作
 fs.close(fp);
});

这里面的

'wx': Like 'w' but fails if the path exists.

在open调用中,第二个参数中所包含的’x’,指的就是要以独占的方式打开指定文件, 如果指定文件已经被别的进程或线程打开了,那么if(err)的判断成立,函数就会直接打印错误信息并返回。

利用隔离标志实现文件的原子操作有一个缺陷,并不是所有的平台都会支持该标志,例如要读取网络硬盘上的文件,那么该标志很可能会被忽略掉。因此使用隔离标志读取跨平台的文件时,隔离作用很可能会失效。

为了弥补这个缺陷,更好的办法是使用目录来实现文件锁。mkdir是标准的POSIX系统调用,根据标准,它的实现必须是原子性的。任何支持该调用的平台都必须保证它执行的原子性,利用这个特性来构建文件锁的话,代码就能得到很好的跨平台支持,即使是读取网络文件,也能保证文件的一致性不会被破坏。接下来我们看看,如何在NodeJS中使用mkdir实现文件锁。

2、mkdir版本的

下面方法是ts的,基于node14的。如果是node版本比较低的话,需要修改mkdir等方法为promise的。

import * as path from "path";
var fs_promise = require("fs/promises");
const { mkdir, writeFile, unlink, rmdir } = fs_promise;

const LOCAL_PB_LOCK_PATH = path.resolve(__dirname, `../config.lock`);

class Lock {
    static hasLock = false;

    public async lock() {
        if (Lock.hasLock) return Promise.reject(`已经有方法处理中`);
        try {
            await mkdir(LOCAL_PB_LOCK_PATH);
            await writeFile(LOCAL_PB_LOCK_PATH + '/' + process.pid, 'lock');
            Lock.hasLock = true;
            return 'success';
        } catch (err) {
            console.warn(`[mock]lock error ${err}`);
            return Promise.reject(`加锁失败`);
        }
    }

    public async unlock() {
        if (!Lock.hasLock) return "success";

        try {
            await unlink(LOCAL_PB_LOCK_PATH + '/' + process.pid);
            await rmdir(LOCAL_PB_LOCK_PATH);
            Lock.hasLock = false;
            return "success";
        } catch (err) {
            console.warn(`unlock error ${err}`);
            return "success";
        }
    }
}

export default new Lock();