var express = require('express') , app = express.createServer() , _ = require('underscore') , fs = require('fs') , mixpanel = require('mixpanel') , exec = require('child_process').exec , spawn = require('child_process').spawn , Stream = require('stream') , config = require('./config.js') , providers = require('./providers.js').list var mpq = new mixpanel.Client('6e6e6b71ed5ada4504c52d915388d73d'); var redis; if (process.env.NODE_ENV == 'production') redis = require('redis-url').connect(process.env.REDISTOGO_URL); else redis = require('redis-url').connect(); // Express config app.set('views', __dirname + '/views'); app.set('view engine', 'jade'); app.use(express.cookieParser()); app.use(express.static(__dirname + '/public')); app.use(express.bodyParser()); // App /* Homepage */ app.get('/', function(req, res) { fs.readFile(__dirname + '/views/index.html', 'utf8', function(err, text){ res.send(text); }); }); app.post('/text', function(req, res) { if (!req.body.number || !req.body.message) { mpq.track('incomplete request'); res.send({success:false,message:'Number and message parameters are required.'}); return; } mpq.track('text', {number: req.body.number, message: req.body.message, ip: req.connection.remoteAddress}); var number = stripPhone(req.body.number); if (number.length < 9 || number.length > 10) { res.send({success:false,message:'Invalid phone number.'}); return; } var ipkey = 'textbelt:ip:' + req.connection.remoteAddress + '_' + dateStr(); var phonekey = 'textbelt:phone:' + number; redis.incr(phonekey, function(err, num) { if (err) { mpq.track('redis fail'); res.send({success:false,message:'Could not validate phone# quota.'}); return; } setTimeout(function() { redis.decr(phonekey, function(err, num) { if (err) { mpq.track('failed to decr phone quota', {number: number}); console.log('*** WARNING failed to decr ' + number); } }); }, 1000*60*3); if (num > 3) { mpq.track('exceeded phone quota'); res.send({success:false,message:'Exceeded quota for this phone number.'}); return; } // now check against ip quota redis.incr(ipkey, function(err, num) { if (err) { mpq.track('redis fail'); res.send({success:false,message:'Could not validate IP quota.'}); return; } /* if (num > 75) { mpq.track('exceeded ip quota'); res.send({success:false,message:'Exceeded quota for this IP address.'}); return; } */ sendText(req.body.number, req.body.message, function(err) { if (err) { mpq.track('sendText failed'); res.send({success:false,message:'Communication with SMS gateway failed.'}); } else { mpq.track('sendText success'); res.send({success:true}); } }); }); }); }); function dateStr() { var today = new Date(); var dd = today.getDate(); var mm = today.getMonth()+1; var yyyy = today.getFullYear(); return mm + '/' + dd + '/' + yyyy; } function stripPhone(phone) { return (phone+'').replace(/\D/g, ''); } function validatePhone(phone) { } function sendText(phone, message, cb) { console.log('txting phone', phone); console.log('msg', message); var done = _.after(providers.length, function() { cb(false); }); _.each(providers, function(provider) { var email = provider.replace('%s', phone); var child = spawn('sendmail', ['-f', 'txt@textbelt.com', email]); child.stdout.on('data', console.log); child.stderr.on('data', console.log); child.on('error', function(data) { mpq.track('sendmail failed', {email: email, data: data}); done(); }); child.on('exit', function(code, signal) { done(); }); child.stdin.write(message + '\n.'); child.stdin.end(); }); } var port = process.env.PORT || 9090; app.listen(port, function() { console.log('Listening on', port); });