reduce to 1 lambda for buzzer
This commit is contained in:
parent
f6fac50adb
commit
573d3a063a
@ -1,20 +1,61 @@
|
|||||||
/**
|
/**
|
||||||
* Simple call box routine
|
* Doorman entrypoint
|
||||||
*
|
|
||||||
* This function is meant for the apartment building callbox
|
|
||||||
* It gives the user a couple of seconds to produce the password
|
|
||||||
* Then dials all the residents to grant manual entry
|
|
||||||
*/
|
*/
|
||||||
const fetch = require('node-fetch');
|
const fetch = require('node-fetch');
|
||||||
|
|
||||||
exports.handler = async function(context, event, callback) {
|
async function getConfig(context, buzzer) {
|
||||||
let twiml = new Twilio.twiml.VoiceResponse();
|
return await fetch(context.DOORMAN_URL + `/api/door/info?buzzer=${buzzer}`)
|
||||||
|
|
||||||
let config = await fetch(context.DOORMAN_URL + `/api/door/info?buzzer=${event.From}`)
|
|
||||||
.then(res => res.json())
|
.then(res => res.json())
|
||||||
.catch(err => {
|
.catch(err => {
|
||||||
return undefined;
|
return undefined;
|
||||||
});
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function notifyDiscord(context, msg, u, optionalJsonStr) {
|
||||||
|
return fetch(context.DOORMAN_URL +
|
||||||
|
`/api/door/notify?discordUser=${u}&msg=${encodeURIComponent(msg)}&json=${encodeURIComponent(optionalJsonStr)}`
|
||||||
|
).catch(err => console.log(err))
|
||||||
|
}
|
||||||
|
|
||||||
|
function notifyAllDiscord(context, config, msg, optionalJsonStr) {
|
||||||
|
return Promise.all(config.discordUsers.map((u) =>
|
||||||
|
notifyDiscord(context, msg, u, optionalJsonStr)
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
|
function doorOpenTwiml(twiml, config) {
|
||||||
|
twiml.play('https://buzzer-2439-prod.twil.io/buzzing_up_boosted.mp3');
|
||||||
|
twiml.play({ digits: config.pressKey }); // configured in doorman what button to click and passed into this function
|
||||||
|
twiml.pause({ length: 1 });
|
||||||
|
twiml.hangup();
|
||||||
|
}
|
||||||
|
|
||||||
|
function dialFallbackTwiml(twiml, config) {
|
||||||
|
let dial = twiml.dial({
|
||||||
|
timeLimit: 20,
|
||||||
|
timeout: 20
|
||||||
|
});
|
||||||
|
|
||||||
|
config.fallbackNumbers.forEach(number => {
|
||||||
|
dial.number(number);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
exports.handler = async function(context, event, callback) {
|
||||||
|
let twiml = new Twilio.twiml.VoiceResponse();
|
||||||
|
|
||||||
|
let config = event.config;
|
||||||
|
|
||||||
|
// get by api or parse it out from query
|
||||||
|
if (!config) {
|
||||||
|
config = await getConfig(context, event.From);
|
||||||
|
} else {
|
||||||
|
try {
|
||||||
|
config = JSON.parse(config);
|
||||||
|
} catch(e) {
|
||||||
|
config = await getConfig(context, event.From);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// reject the call if this is not configured
|
// reject the call if this is not configured
|
||||||
if (!config || !config.door) {
|
if (!config || !config.door) {
|
||||||
@ -23,31 +64,25 @@ exports.handler = async function(context, event, callback) {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
let configQuery = `config=${encodeURIComponent(JSON.stringify(config))}`;
|
// TODO: await on this?
|
||||||
|
// let users know someone is currently buzzing, and allow unlock by discord user
|
||||||
// let user know someone is currently buzzing, and allow unlock by discord user
|
|
||||||
let msg = `🔔 Someone is dialing right now @ Door "${config.door}" [Click to unlock](${context.DOORMAN_URL}/api/door/auth?door=${config.door}&key=`;
|
let msg = `🔔 Someone is dialing right now @ Door "${config.door}" [Click to unlock](${context.DOORMAN_URL}/api/door/auth?door=${config.door}&key=`;
|
||||||
config.discordUsers.forEach((u) => {
|
let discordNotifs = Promise.all(config.discordUsers.map((u) =>
|
||||||
// unlock me by discord userid
|
notifyDiscord(context, msg + u + ')', u)
|
||||||
fetch(context.DOORMAN_URL + `/api/door/notify?discordUser=${u}&msg=${encodeURIComponent(msg + u + ')')}`);
|
));
|
||||||
});
|
|
||||||
|
|
||||||
// poll Doorman, to see if we should unlock
|
// poll Doorman, to see if we should unlock
|
||||||
const interval = setInterval(() => {
|
const interval = setInterval(() => {
|
||||||
fetch(context.DOORMAN_URL + `/api/door/status?door=${config.door}`)
|
fetch(context.DOORMAN_URL + `/api/door/status?door=${config.door}`)
|
||||||
.then(async res => {
|
.then(async res => {
|
||||||
// handle the case where doorman is explictly rejecting the buzzer
|
if (res.status === 200) {
|
||||||
if (res.status === 410) {
|
|
||||||
clearInterval(interval);
|
clearInterval(interval);
|
||||||
twiml.redirect(`/text-me?method=doorman-time-lock&${configQuery}`);
|
// put this before awaiting notify
|
||||||
callback(null, twiml);
|
// so it would be appended above any dial Verb from fallback
|
||||||
}
|
doorOpenTwiml(twiml, config);
|
||||||
|
|
||||||
// we got the successful unlock
|
|
||||||
else if (res.status === 200) {
|
|
||||||
clearInterval(interval);
|
|
||||||
const body = await res.json();
|
const body = await res.json();
|
||||||
twiml.redirect(`/door-open?fingerprint=${encodeURIComponent(JSON.stringify(body.fingerprint))}&${configQuery}`);
|
await notifyAllDiscord(context, config, `🔓 Doorman buzzed someone up @ Door "${config.door}"`, JSON.stringify(body.fingerprint));
|
||||||
callback(null, twiml);
|
callback(null, twiml);
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
@ -55,8 +90,14 @@ exports.handler = async function(context, event, callback) {
|
|||||||
}, 500);
|
}, 500);
|
||||||
|
|
||||||
// redirect to call after 6s
|
// redirect to call after 6s
|
||||||
setTimeout(() => {
|
setTimeout(async () => {
|
||||||
twiml.redirect(`/call-residents?${configQuery}`);
|
dialFallbackTwiml(twiml, config);
|
||||||
|
await notifyAllDiscord(
|
||||||
|
context,
|
||||||
|
config,
|
||||||
|
`📞 Somebody buzzed the door and it dialed through to fallback phone numbers @ Door "${config.door}"`
|
||||||
|
);
|
||||||
|
|
||||||
callback(null, twiml);
|
callback(null, twiml);
|
||||||
}, 6000);
|
}, 6000);
|
||||||
};
|
};
|
||||||
@ -1,23 +0,0 @@
|
|||||||
/**
|
|
||||||
* Fallback behavior, if the code is wrong or unspecified, then we should dial the fallback numbers
|
|
||||||
*/
|
|
||||||
exports.handler = function(context, event, callback) {
|
|
||||||
let twiml = new Twilio.twiml.VoiceResponse();
|
|
||||||
|
|
||||||
let config = JSON.parse(event.config);
|
|
||||||
console.log(config);
|
|
||||||
let configQuery = `config=${encodeURIComponent(JSON.stringify(config))}`;
|
|
||||||
|
|
||||||
// If no valid answer after timeout, dial all residents until someone picks up
|
|
||||||
let dial = twiml.dial({
|
|
||||||
action: `/text-me?Method=call&${configQuery}`,
|
|
||||||
timeLimit: 20,
|
|
||||||
timeout: 20
|
|
||||||
});
|
|
||||||
|
|
||||||
config.fallbackNumbers.forEach(number => {
|
|
||||||
dial.number(number);
|
|
||||||
});
|
|
||||||
|
|
||||||
return callback(null, twiml);
|
|
||||||
}
|
|
||||||
@ -1,18 +0,0 @@
|
|||||||
/**
|
|
||||||
* Automatically open the door
|
|
||||||
*/
|
|
||||||
exports.handler = function(context, event, callback) {
|
|
||||||
let twiml = new Twilio.twiml.VoiceResponse();
|
|
||||||
|
|
||||||
let config = JSON.parse(event.config);
|
|
||||||
let configQuery = `config=${encodeURIComponent(JSON.stringify(config))}`;
|
|
||||||
|
|
||||||
let passAlong = `fingerprint=${encodeURIComponent(event.fingerprint)}&${configQuery}`;
|
|
||||||
|
|
||||||
twiml.play('https://buzzer-2439-prod.twil.io/buzzing_up_boosted.mp3');
|
|
||||||
twiml.play({ digits: config.pressKey }); // configured in doorman what button to click and passed into this function
|
|
||||||
twiml.pause({ length: 1 });
|
|
||||||
twiml.redirect(`/text-me?Method=doorman&${passAlong}`);
|
|
||||||
|
|
||||||
callback(null, twiml);
|
|
||||||
};
|
|
||||||
@ -1,36 +0,0 @@
|
|||||||
/**
|
|
||||||
* Send a NTFY message with an update of what happened. If the password was used or not...
|
|
||||||
*/
|
|
||||||
|
|
||||||
const fetch = require('node-fetch');
|
|
||||||
|
|
||||||
exports.handler = async function(context, event, callback) {
|
|
||||||
let twiml = new Twilio.twiml.VoiceResponse();
|
|
||||||
let config = JSON.parse(event.config);
|
|
||||||
|
|
||||||
// should be a list of strings representing user ids in discord
|
|
||||||
let discordUsers = config.discordUsers || [];
|
|
||||||
|
|
||||||
let msg;
|
|
||||||
let optionalJson = event.fingerprint;
|
|
||||||
|
|
||||||
if (event.Method == 'doorman') {
|
|
||||||
msg = '🔓 Doorman buzzed someone up';
|
|
||||||
} else if (event.Method == 'doorman-time-lock') {
|
|
||||||
msg = '🔒 Doorman rejected a buzzer call due to time restriction';
|
|
||||||
} else if (event.Method == 'call') {
|
|
||||||
msg = '📞 Somebody buzzed the door and it dialed through to a fallback phone number';
|
|
||||||
}
|
|
||||||
|
|
||||||
msg += ` @ Door "${config.door}"`;
|
|
||||||
|
|
||||||
let promises = discordUsers.map((u) =>
|
|
||||||
fetch(context.DOORMAN_URL + `/api/door/notify?discordUser=${u}&msg=${msg}&json=${optionalJson}`)
|
|
||||||
.catch(err => console.log(err))
|
|
||||||
);
|
|
||||||
|
|
||||||
await Promise.all(promises)
|
|
||||||
.catch (e => console.log(e));
|
|
||||||
|
|
||||||
return callback(null, twiml);
|
|
||||||
};
|
|
||||||
Loading…
x
Reference in New Issue
Block a user