Blame view

node_modules/bonjour/lib/mdns-server.js 3.1 KB
aaac7fed   liuqimichale   add
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
  'use strict'
  
  var multicastdns = require('multicast-dns')
  var dnsEqual = require('dns-equal')
  var flatten = require('array-flatten')
  var deepEqual = require('deep-equal')
  
  module.exports = Server
  
  function Server (opts) {
    this.mdns = multicastdns(opts)
    this.mdns.setMaxListeners(0)
    this.registry = {}
    this.mdns.on('query', this._respondToQuery.bind(this))
  }
  
  Server.prototype.register = function (records) {
    var self = this
  
    if (Array.isArray(records)) records.forEach(register)
    else register(records)
  
    function register (record) {
      var subRegistry = self.registry[record.type]
      if (!subRegistry) subRegistry = self.registry[record.type] = []
      else if (subRegistry.some(isDuplicateRecord(record))) return
      subRegistry.push(record)
    }
  }
  
  Server.prototype.unregister = function (records) {
    var self = this
  
    if (Array.isArray(records)) records.forEach(unregister)
    else unregister(records)
  
    function unregister (record) {
      var type = record.type
      if (!(type in self.registry)) return
      self.registry[type] = self.registry[type].filter(function (r) {
        return r.name !== record.name
      })
    }
  }
  
  Server.prototype._respondToQuery = function (query) {
    var self = this
    query.questions.forEach(function (question) {
      var type = question.type
      var name = question.name
  
      // generate the answers section
      var answers = type === 'ANY'
        ? flatten.depth(Object.keys(self.registry).map(self._recordsFor.bind(self, name)), 1)
        : self._recordsFor(name, type)
  
      if (answers.length === 0) return
  
      // generate the additionals section
      var additionals = []
      if (type !== 'ANY') {
        answers.forEach(function (answer) {
          if (answer.type !== 'PTR') return
          additionals = additionals
            .concat(self._recordsFor(answer.data, 'SRV'))
            .concat(self._recordsFor(answer.data, 'TXT'))
        })
  
        // to populate the A and AAAA records, we need to get a set of unique
        // targets from the SRV record
        additionals
          .filter(function (record) {
            return record.type === 'SRV'
          })
          .map(function (record) {
            return record.data.target
          })
          .filter(unique())
          .forEach(function (target) {
            additionals = additionals
              .concat(self._recordsFor(target, 'A'))
              .concat(self._recordsFor(target, 'AAAA'))
          })
      }
  
      self.mdns.respond({ answers: answers, additionals: additionals }, function (err) {
        if (err) throw err // TODO: Handle this (if no callback is given, the error will be ignored)
      })
    })
  }
  
  Server.prototype._recordsFor = function (name, type) {
    if (!(type in this.registry)) return []
  
    return this.registry[type].filter(function (record) {
      var _name = ~name.indexOf('.') ? record.name : record.name.split('.')[0]
      return dnsEqual(_name, name)
    })
  }
  
  function isDuplicateRecord (a) {
    return function (b) {
      return a.type === b.type &&
        a.name === b.name &&
        deepEqual(a.data, b.data)
    }
  }
  
  function unique () {
    var set = []
    return function (obj) {
      if (~set.indexOf(obj)) return false
      set.push(obj)
      return true
    }
  }