仅在经过一段时间后才允许调用函数

时间:2017-03-29 19:52:47

标签: javascript node.js timer groupme

要提供上下文,以下是我试图解决的问题:

我和我的朋友一起制作了一个giphy机器人。通过在邮件中输入/giphy [terms],它会自动发布[terms]的最高结果。我的朋友,他们是吵闹的混蛋,很快就开始滥用它来垃圾邮件群聊。我想做的是防止这种情况,只允许每分钟调用一次postMessage函数。

我尝试了什么:

  • 使用setTimeout(),它不能完全按照我的意愿行事,因为它只会在参数中指定的时间量过后调用该函数。据我所知,这会导致从调用机器人时起消息延迟,但实际上并不会阻止僵尸程序在此时接受新的postMessage()调用。
  • 使用setInterval(),这只会导致以一定的间隔永久调用该函数。

我认为可行的方法:

现在,我正在使用两个.js文件。

Index.js

var http, director, cool, bot, router, server, port;

http        = require('http');
director    = require('director');
bot         = require('./bot.js');

router = new director.http.Router({
  '/' : {
    post: bot.respond,
    get: ping
  }
});

server = http.createServer(function (req, res) {
  req.chunks = [];
  req.on('data', function (chunk) {
    req.chunks.push(chunk.toString());
  });

  router.dispatch(req, res, function(err) {
    res.writeHead(err.status, {"Content-Type": "text/plain"});
    res.end(err.message);
  });
});

port = Number(process.env.PORT || 5000);
server.listen(port);

function ping() {
  this.res.writeHead(200);
  this.res.end("This is my giphy side project!");
}

Bot.js

var HTTPS = require('https');
var botID = process.env.BOT_ID;
var giphy = require('giphy-api')();

function respond() {
  var request = JSON.parse(this.req.chunks[0]);
  var giphyRegex = /^\/giphy (.*)$/;
  var botMessage = giphyRegex.exec(request.text);
  var offset = Math.floor(Math.random() * 10);

  if(request.text && giphyRegex.test(request.text) && botMessage != null) {
    this.res.writeHead(200);
    giphy.search({
      q: botMessage[1],
      rating: 'pg-13'
    }, function (err, res) {
      try {
        postMessage(res.data[offset].images.downsized.url);
      } catch (err) {
        postMessage("There is no gif of that.");
      }
    });
    this.res.end();
  } else {
    this.res.writeHead(200);
    this.res.end();
  }

function postMessage(phrase) {
  var botResponse, options, body, botReq;
  botResponse = phrase;

  options = {
    hostname: 'api.groupme.com',
    path: '/v3/bots/post',
    method: 'POST'
  };

  body = {
    "bot_id" : botID,
    "text" : botResponse
  };

  botReq = HTTPS.request(options, function(res) {
      if(res.statusCode == 202) {
      } else {
        console.log('Rejecting bad status code: ' + res.statusCode);
      }
  });

  botReq.on('error', function(err) {
    console.log('Error posting message: '  + JSON.stringify(err));
  });

  botReq.on('timeout', function(err) {
    console.log('Timeout posting message: '  + JSON.stringify(err));
  });

  botReq.end(JSON.stringify(body));
}
exports.respond = respond;

基本上,我想知道在哪里实现我想象的计时器的理想位置。好像我想让它只在一分钟后听/giphy [terms],而不是等一分钟发布。

我的问题:

  • 最好的方法是在response()函数上设置一个计时器,从那以后它每分钟只会实际解析一次传入的信息吗?是否有一个更优雅的地方放这个?

  • 计时器应如何处理该功能?我认为我不能每分钟运行一次response(),因为这似乎意味着它每分钟只会解析一次来自GroupMe API的传入json,所以它可能会错过我想要它的传入消息捕获。

1 个答案:

答案 0 :(得分:1)

存储请求的时间,然后使用它来查看后续请求是否应该被忽略(如果这些请求执行得很快)。

var waitTime = 10*1000; // 10 s in millis 
var lastRequestTime = null;
function respond() {
  if(lastRequestTime){
    var now = new Date();
    if(now.getTime() - lastRequestTime.getTime() <= waitTime){
        this.res.writeHead(200);
        this.res.end("You have to wait "+waitTime/1000+" seconds.");
        return;
    } 
  }
  lastRequestTime = new Date();
  postMessage();
}