| 
									
										
										
										
											2016-10-17 17:40:55 -06:00
										 |  |  | 'use strict'; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-14 15:26:19 -06:00
										 |  |  | module.exports.create = function (deps, conf) { | 
					
						
							|  |  |  |   var PromiseA = deps.PromiseA; | 
					
						
							|  |  |  |   var request = PromiseA.promisify(require('request')); | 
					
						
							|  |  |  |   var OAUTH3 = require('../packages/assets/org.oauth3'); | 
					
						
							|  |  |  |   require('../packages/assets/org.oauth3/oauth3.dns.js'); | 
					
						
							| 
									
										
										
										
											2017-09-14 18:28:49 -06:00
										 |  |  |   OAUTH3._hooks = require('../packages/assets/org.oauth3/oauth3.node.storage.js'); | 
					
						
							| 
									
										
										
										
											2016-10-17 17:40:55 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-14 15:26:19 -06:00
										 |  |  |   function dnsType(addr) { | 
					
						
							|  |  |  |     if (/^\d+\.\d+\.\d+\.\d+$/.test(addr)) { | 
					
						
							|  |  |  |       return 'A'; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (-1 !== addr.indexOf(':') && /^[a-f:\.\d]+$/i.test(addr)) { | 
					
						
							|  |  |  |       return 'AAAA'; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-10-17 17:40:55 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-19 13:18:22 -06:00
										 |  |  |   async function getSession() { | 
					
						
							|  |  |  |     var sessions = await deps.storage.owners.all(); | 
					
						
							|  |  |  |     var session = sessions.filter(function (sess) { | 
					
						
							|  |  |  |       return sess.token.scp.indexOf('dns') >= 0; | 
					
						
							|  |  |  |     })[0]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!session) { | 
					
						
							|  |  |  |       throw new Error('no sessions with DNS grants'); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // The OAUTH3 library stores some things on the root session object that we usually
 | 
					
						
							|  |  |  |     // just leave inside the token, but we need to pull those out before we use it here
 | 
					
						
							|  |  |  |     session.provider_uri = session.provider_uri || session.token.provider_uri || session.token.iss; | 
					
						
							|  |  |  |     session.client_uri = session.client_uri || session.token.azp; | 
					
						
							|  |  |  |     session.scope = session.scope || session.token.scp; | 
					
						
							|  |  |  |     return session; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   async function setDeviceAddress(addr) { | 
					
						
							|  |  |  |     var session = await getSession(); | 
					
						
							|  |  |  |     var directives = await OAUTH3.discover(session.token.aud); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Set the address of the device to our public address.
 | 
					
						
							|  |  |  |     await request({ | 
					
						
							|  |  |  |       url: directives.api+'/api/com.daplie.domains/acl/devices/' + conf.device.hostname | 
					
						
							|  |  |  |     , method: 'POST' | 
					
						
							|  |  |  |     , headers: { | 
					
						
							|  |  |  |         'Authorization': 'Bearer ' + session.refresh_token | 
					
						
							|  |  |  |       , 'Accept': 'application/json; charset=utf-8' | 
					
						
							| 
									
										
										
										
											2017-09-14 15:26:19 -06:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2017-09-19 13:18:22 -06:00
										 |  |  |     , json: { | 
					
						
							|  |  |  |         addresses: [ | 
					
						
							|  |  |  |           { value: addr, type:  dnsType(addr) } | 
					
						
							|  |  |  |         ] | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2016-10-17 17:40:55 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-19 13:18:22 -06:00
										 |  |  |     // Then update all of the records attached to our hostname, first removing the old records
 | 
					
						
							|  |  |  |     // to remove the reference to the old address, then creating new records for the same domains
 | 
					
						
							|  |  |  |     // using our new address.
 | 
					
						
							|  |  |  |     var allDns = OAUTH3.api(directives.api, {session: session, api: 'dns.list'}); | 
					
						
							|  |  |  |     var ourDomains = allDns.filter(function (record) { | 
					
						
							|  |  |  |       return record.device === conf.device.hostname; | 
					
						
							|  |  |  |     }).map(function (record) { | 
					
						
							|  |  |  |       var zoneSplit = record.zone.split('.'); | 
					
						
							|  |  |  |       return { | 
					
						
							|  |  |  |         tld: zoneSplit.slice(1).join('.') | 
					
						
							|  |  |  |       , sld: zoneSplit[0] | 
					
						
							|  |  |  |       , sub: record.host.slice(0, -(record.zone.length + 1)) | 
					
						
							|  |  |  |       }; | 
					
						
							| 
									
										
										
										
											2016-10-17 17:40:55 -06:00
										 |  |  |     }); | 
					
						
							| 
									
										
										
										
											2017-09-19 13:18:22 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  |     var common = { | 
					
						
							|  |  |  |       api: 'devices.detach' | 
					
						
							|  |  |  |     , session: session | 
					
						
							|  |  |  |     , device: conf.device.hostname | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  |     await PromiseA.all(ourDomains.map(function (record) { | 
					
						
							|  |  |  |       return OAUTH3.api(directives.api, Object.assign({}, common, record)); | 
					
						
							|  |  |  |     })); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     common = { | 
					
						
							|  |  |  |       api: 'devices.attach' | 
					
						
							|  |  |  |     , session: session | 
					
						
							|  |  |  |     , device: conf.device.hostname | 
					
						
							|  |  |  |     , ip: addr | 
					
						
							|  |  |  |     , ttl: 300 | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  |     await PromiseA.all(ourDomains.map(function (record) { | 
					
						
							|  |  |  |       return OAUTH3.api(directives.api, Object.assign({}, common, record)); | 
					
						
							|  |  |  |     })); | 
					
						
							| 
									
										
										
										
											2017-09-14 15:26:19 -06:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-10-17 17:40:55 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-19 13:18:22 -06:00
										 |  |  |   async function getDeviceAddresses() { | 
					
						
							|  |  |  |     var session = await getSession(); | 
					
						
							|  |  |  |     var directives = await OAUTH3.discover(session.token.aud); | 
					
						
							| 
									
										
										
										
											2016-10-17 17:40:55 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-19 13:18:22 -06:00
										 |  |  |     var result = await request({ | 
					
						
							|  |  |  |       url: directives.api+'/api/org.oauth3.dns/acl/devices' | 
					
						
							|  |  |  |     , method: 'GET' | 
					
						
							|  |  |  |     , headers: { | 
					
						
							|  |  |  |         'Authorization': 'Bearer ' + session.refresh_token | 
					
						
							|  |  |  |       , 'Accept': 'application/json; charset=utf-8' | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     , json: true | 
					
						
							| 
									
										
										
										
											2016-10-17 17:40:55 -06:00
										 |  |  |     }); | 
					
						
							| 
									
										
										
										
											2017-09-19 13:18:22 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (!result.body) { | 
					
						
							|  |  |  |       throw new Error('No response body in request for device addresses'); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (result.body.error) { | 
					
						
							|  |  |  |       throw Object.assign(new Error('error getting device list'), result.body.error); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     var dev = result.body.devices.filter(function (dev) { | 
					
						
							|  |  |  |       return dev.name === conf.device.hostname; | 
					
						
							|  |  |  |     })[0]; | 
					
						
							|  |  |  |     return (dev || {}).addresses || []; | 
					
						
							| 
									
										
										
										
											2017-09-14 15:26:19 -06:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   var publicAddress; | 
					
						
							| 
									
										
										
										
											2017-09-19 13:18:22 -06:00
										 |  |  |   async function recheckPubAddr() { | 
					
						
							| 
									
										
										
										
											2017-09-14 15:26:19 -06:00
										 |  |  |     if (!conf.ddns.enabled) { | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-19 13:18:22 -06:00
										 |  |  |     var session = await getSession(); | 
					
						
							|  |  |  |     var directives = await OAUTH3.discover(session.token.aud); | 
					
						
							|  |  |  |     var addr = await deps.loopback.checkPublicAddr(directives.api); | 
					
						
							| 
									
										
										
										
											2016-10-17 17:40:55 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-19 13:18:22 -06:00
										 |  |  |     if (publicAddress === addr) { | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (conf.debug) { | 
					
						
							|  |  |  |       console.log('previous public address',publicAddress, 'does not match current public address', addr); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     await setDeviceAddress(addr); | 
					
						
							|  |  |  |     publicAddress = addr; | 
					
						
							| 
									
										
										
										
											2017-09-14 15:26:19 -06:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-10-17 17:40:55 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-14 18:28:49 -06:00
										 |  |  |   recheckPubAddr(); | 
					
						
							| 
									
										
										
										
											2017-09-14 15:26:19 -06:00
										 |  |  |   setInterval(recheckPubAddr, 5*60*1000); | 
					
						
							| 
									
										
										
										
											2016-10-17 17:40:55 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-14 15:26:19 -06:00
										 |  |  |   return { | 
					
						
							|  |  |  |     setDeviceAddress:   setDeviceAddress | 
					
						
							|  |  |  |   , getDeviceAddresses: getDeviceAddresses | 
					
						
							|  |  |  |   , recheckPubAddr:     recheckPubAddr | 
					
						
							| 
									
										
										
										
											2016-10-17 17:40:55 -06:00
										 |  |  |   }; | 
					
						
							| 
									
										
										
										
											2017-09-14 15:26:19 -06:00
										 |  |  | }; |