"use strict"; var __create = Object.create; var __defProp = Object.defineProperty; var __getOwnPropDesc = Object.getOwnPropertyDescriptor; var __getOwnPropNames = Object.getOwnPropertyNames; var __getProtoOf = Object.getPrototypeOf; var __hasOwnProp = Object.prototype.hasOwnProperty; var __copyProps = (to, from, except, desc) => { if (from && typeof from === "object" || typeof from === "function") { for (let key of __getOwnPropNames(from)) if (!__hasOwnProp.call(to, key) && key !== except) __defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable }); } return to; }; var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps( isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target, mod )); var fs = __toESM(require("fs")); var os = __toESM(require("os")); var path = __toESM(require("path")); const SYSTEMD_SERVICE_NAME = "iobroker-ds18b20-remote.service"; const files = { 'common.js': 'InVzZSBzdHJpY3QiOwp2YXIgX19jcmVhdGUgPSBPYmplY3QuY3JlYXRlOwp2YXIgX19kZWZQcm9wID0gT2JqZWN0LmRlZmluZVByb3BlcnR5Owp2YXIgX19nZXRPd25Qcm9wRGVzYyA9IE9iamVjdC5nZXRPd25Qcm9wZXJ0eURlc2NyaXB0b3I7CnZhciBfX2dldE93blByb3BOYW1lcyA9IE9iamVjdC5nZXRPd25Qcm9wZXJ0eU5hbWVzOwp2YXIgX19nZXRQcm90b09mID0gT2JqZWN0LmdldFByb3RvdHlwZU9mOwp2YXIgX19oYXNPd25Qcm9wID0gT2JqZWN0LnByb3RvdHlwZS5oYXNPd25Qcm9wZXJ0eTsKdmFyIF9fZXhwb3J0ID0gKHRhcmdldCwgYWxsKSA9PiB7CiAgZm9yICh2YXIgbmFtZSBpbiBhbGwpCiAgICBfX2RlZlByb3AodGFyZ2V0LCBuYW1lLCB7IGdldDogYWxsW25hbWVdLCBlbnVtZXJhYmxlOiB0cnVlIH0pOwp9Owp2YXIgX19jb3B5UHJvcHMgPSAodG8sIGZyb20sIGV4Y2VwdCwgZGVzYykgPT4gewogIGlmIChmcm9tICYmIHR5cGVvZiBmcm9tID09PSAib2JqZWN0IiB8fCB0eXBlb2YgZnJvbSA9PT0gImZ1bmN0aW9uIikgewogICAgZm9yIChsZXQga2V5IG9mIF9fZ2V0T3duUHJvcE5hbWVzKGZyb20pKQogICAgICBpZiAoIV9faGFzT3duUHJvcC5jYWxsKHRvLCBrZXkpICYmIGtleSAhPT0gZXhjZXB0KQogICAgICAgIF9fZGVmUHJvcCh0bywga2V5LCB7IGdldDogKCkgPT4gZnJvbVtrZXldLCBlbnVtZXJhYmxlOiAhKGRlc2MgPSBfX2dldE93blByb3BEZXNjKGZyb20sIGtleSkpIHx8IGRlc2MuZW51bWVyYWJsZSB9KTsKICB9CiAgcmV0dXJuIHRvOwp9Owp2YXIgX190b0VTTSA9IChtb2QsIGlzTm9kZU1vZGUsIHRhcmdldCkgPT4gKHRhcmdldCA9IG1vZCAhPSBudWxsID8gX19jcmVhdGUoX19nZXRQcm90b09mKG1vZCkpIDoge30sIF9fY29weVByb3BzKAogIGlzTm9kZU1vZGUgfHwgIW1vZCB8fCAhbW9kLl9fZXNNb2R1bGUgPyBfX2RlZlByb3AodGFyZ2V0LCAiZGVmYXVsdCIsIHsgdmFsdWU6IG1vZCwgZW51bWVyYWJsZTogdHJ1ZSB9KSA6IHRhcmdldCwKICBtb2QKKSk7CnZhciBfX3RvQ29tbW9uSlMgPSAobW9kKSA9PiBfX2NvcHlQcm9wcyhfX2RlZlByb3Aoe30sICJfX2VzTW9kdWxlIiwgeyB2YWx1ZTogdHJ1ZSB9KSwgbW9kKTsKdmFyIGNvbW1vbl9leHBvcnRzID0ge307Cl9fZXhwb3J0KGNvbW1vbl9leHBvcnRzLCB7CiAgUkVNT1RFX1BST1RPQ09MX1ZFUlNJT046ICgpID0+IFJFTU9URV9QUk9UT0NPTF9WRVJTSU9OLAogIGRlY3J5cHQ6ICgpID0+IGRlY3J5cHQsCiAgZW5jcnlwdDogKCkgPT4gZW5jcnlwdAp9KTsKbW9kdWxlLmV4cG9ydHMgPSBfX3RvQ29tbW9uSlMoY29tbW9uX2V4cG9ydHMpOwp2YXIgY3J5cHRvID0gX190b0VTTShyZXF1aXJlKCJjcnlwdG8iKSk7CmNvbnN0IFJFTU9URV9QUk9UT0NPTF9WRVJTSU9OID0gMzsKY29uc3QgSVZfTEVOR1RIID0gMTY7CmZ1bmN0aW9uIGVuY3J5cHQodGV4dCwga2V5KSB7CiAgY29uc3QgaXYgPSBjcnlwdG8ucmFuZG9tQnl0ZXMoSVZfTEVOR1RIKTsKICBjb25zdCBjaXBoZXIgPSBjcnlwdG8uY3JlYXRlQ2lwaGVyaXYoImFlcy0yNTYtY2JjIiwga2V5LCBpdik7CiAgbGV0IGVuY3J5cHRlZCA9IGNpcGhlci51cGRhdGUodGV4dCk7CiAgZW5jcnlwdGVkID0gQnVmZmVyLmNvbmNhdChbZW5jcnlwdGVkLCBjaXBoZXIuZmluYWwoKV0pOwogIHJldHVybiBpdi50b1N0cmluZygiaGV4IikgKyAiOiIgKyBlbmNyeXB0ZWQudG9TdHJpbmcoImhleCIpOwp9CmZ1bmN0aW9uIGRlY3J5cHQodGV4dCwga2V5KSB7CiAgY29uc3QgdGV4dFBhcnRzID0gdGV4dC5zcGxpdCgiOiIpOwogIGNvbnN0IGl2ID0gQnVmZmVyLmZyb20odGV4dFBhcnRzLnNoaWZ0KCksICJoZXgiKTsKICBjb25zdCBlbmNyeXB0ZWRUZXh0ID0gQnVmZmVyLmZyb20odGV4dFBhcnRzLmpvaW4oIjoiKSwgImhleCIpOwogIGNvbnN0IGRlY2lwaGVyID0gY3J5cHRvLmNyZWF0ZURlY2lwaGVyaXYoImFlcy0yNTYtY2JjIiwga2V5LCBpdik7CiAgbGV0IGRlY3J5cHRlZCA9IGRlY2lwaGVyLnVwZGF0ZShlbmNyeXB0ZWRUZXh0KTsKICBkZWNyeXB0ZWQgPSBCdWZmZXIuY29uY2F0KFtkZWNyeXB0ZWQsIGRlY2lwaGVyLmZpbmFsKCldKTsKICByZXR1cm4gZGVjcnlwdGVkLnRvU3RyaW5nKCk7Cn0KLy8gQW5ub3RhdGUgdGhlIENvbW1vbkpTIGV4cG9ydCBuYW1lcyBmb3IgRVNNIGltcG9ydCBpbiBub2RlOgowICYmIChtb2R1bGUuZXhwb3J0cyA9IHsKICBSRU1PVEVfUFJPVE9DT0xfVkVSU0lPTiwKICBkZWNyeXB0LAogIGVuY3J5cHQKfSk7Ci8vIyBzb3VyY2VNYXBwaW5nVVJMPWNvbW1vbi5qcy5tYXAK', 'ds18b20-remote-client.js': '"use strict";
var __create = Object.create;
var __defProp = Object.defineProperty;
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
var __getOwnPropNames = Object.getOwnPropertyNames;
var __getProtoOf = Object.getPrototypeOf;
var __hasOwnProp = Object.prototype.hasOwnProperty;
var __copyProps = (to, from, except, desc) => {
  if (from && typeof from === "object" || typeof from === "function") {
    for (let key of __getOwnPropNames(from))
      if (!__hasOwnProp.call(to, key) && key !== except)
        __defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
  }
  return to;
};
var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(
  isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target,
  mod
));
var import_util = require("util");
var import_net = require("net");
var fs = __toESM(require("fs"));
var os = __toESM(require("os"));
var import_logger = require("./logger");
var import_common = require("./common");
const readDir = (0, import_util.promisify)(fs.readdir);
const readFile = (0, import_util.promisify)(fs.readFile);
const ENV_KEYS = [
  "ADAPTER_HOST",
  "ADAPTER_KEY",
  "ADAPTER_PORT",
  "DEBUG",
  "SYSTEM_ID",
  "W1_DEVICES_PATH"
];
class Ds18b20Remote {
  constructor() {
    this.reconnectTimeout = null;
    this.shouldExit = false;
    this.recvData = "";
    this.connect = this.connect.bind(this);
    this.exit = this.exit.bind(this);
    this.onClose = this.onClose.bind(this);
    this.onData = this.onData.bind(this);
    this.onError = this.onError.bind(this);
    this.onConnect = this.onConnect.bind(this);
    this.log = new import_logger.Logger();
    this.log.log("- ioBroker.ds18b20 remote client -");
    this.readDotEnv();
    if (process.env.SYSTEM_ID) {
      this.systemId = process.env.SYSTEM_ID.trim();
    } else {
      this.systemId = os.hostname();
      this.log.warn(`Using the hostname ${this.systemId} as system ID. Please set SYSTEM_ID to a unique value.`);
    }
    this.log.debug(`systemId`, this.systemId);
    if (process.env.ADAPTER_PORT) {
      try {
        this.adapterPort = parseInt(process.env.ADAPTER_PORT, 10);
      } catch (err) {
        this.log.error(`Invalid ADAPTER_PORT!`, err);
        process.exit(1);
      }
    } else {
      this.adapterPort = 1820;
    }
    this.log.debug(`adapterPort`, this.adapterPort);
    this.adapterHost = (process.env.ADAPTER_HOST || "").trim();
    if (this.adapterHost.length <= 0) {
      this.log.error(`No ADAPTER_HOST given!`);
      process.exit(1);
    }
    this.log.debug(`adapterHost`, this.adapterHost);
    this.adapterKey = Buffer.from(process.env.ADAPTER_KEY || "", "hex");
    if (this.adapterKey.length !== 32) {
      this.log.error(`ADAPTER_KEY is no valid key!`);
      process.exit(1);
    }
    this.log.debug(`adapterKey`, this.adapterKey);
    this.w1DevicesPath = process.env.W1_DEVICES_PATH || "/sys/bus/w1/devices";
    if (!fs.existsSync(this.w1DevicesPath)) {
      this.log.error(`The 1-wire devices path ${this.w1DevicesPath} does not exist!`);
      process.exit(1);
    }
    this.log.debug(`w1DevicesPath`, this.w1DevicesPath);
    process.on("SIGINT", this.exit);
    process.on("SIGTERM", this.exit);
    this.socket = new import_net.Socket();
    this.socket.on("close", this.onClose);
    this.socket.on("data", this.onData);
    this.socket.on("error", this.onError);
    this.socket.on("connect", this.onConnect);
    this.connect();
  }
  connect() {
    if (this.reconnectTimeout) {
      clearTimeout(this.reconnectTimeout);
      this.reconnectTimeout = null;
    }
    if (this.shouldExit) {
      return;
    }
    this.log.info(`Connecting to ${this.adapterHost}:${this.adapterPort} ...`);
    this.socket.connect({
      host: this.adapterHost,
      port: this.adapterPort
    });
  }
  onConnect() {
    this.log.info(`Connected with adapter`);
    if (this.reconnectTimeout) {
      clearTimeout(this.reconnectTimeout);
    }
    this.reconnectTimeout = null;
  }
  onData(data) {
    this.recvData += data.toString();
    let idx = this.recvData.indexOf("\n");
    while (idx > 0) {
      const raw = this.recvData.slice(0, idx);
      this.recvData = this.recvData.slice(idx + 1);
      this.handleSocketData(raw);
      idx = this.recvData.indexOf("\n");
    }
  }
  async handleSocketData(raw) {
    let data;
    try {
      const dataStr = (0, import_common.decrypt)(raw, this.adapterKey);
      data = JSON.parse(dataStr);
    } catch (err) {
      this.log.warn(`Decrypt of data failed! ${err.toString()}`);
      this.socket.end();
      return;
    }
    this.log.debug("message from adapter:", data);
    switch (data.cmd) {
      case "clientInfo":
        if (data.protocolVersion !== import_common.REMOTE_PROTOCOL_VERSION) {
          this.log.warn(`Protocol version ${data.protocolVersion} from the adapter does not match the remote client protocol version ${import_common.REMOTE_PROTOCOL_VERSION}! Please reinstall the remote client.`);
        }
        this.log.info("Sending client info to the adapter");
        this.send({
          cmd: "clientInfo",
          protocolVersion: import_common.REMOTE_PROTOCOL_VERSION,
          systemId: this.systemId
        });
        break;
      case "read":
        if (!data.address) {
          this.log.warn(`Got read command without address from adapter!`);
          return;
        }
        let raw2;
        try {
          raw2 = await readFile(`${this.w1DevicesPath}/${data.address}/w1_slave`, "utf8");
          this.log.debug(`Read from file ${this.w1DevicesPath}/${data.address}/w1_slave:`, raw2);
        } catch (err) {
          this.log.warn(`Read from file ${this.w1DevicesPath}/${data.address}/w1_slave failed! ${err.toString()}`);
          this.log.debug(err);
          raw2 = "";
        }
        await this.send({
          cmd: "read",
          address: data.address,
          ts: data.ts,
          raw: raw2
        });
        break;
      case "search":
        try {
          const files = await readDir(this.w1DevicesPath);
          const proms = [];
          for (const file of files) {
            if (file.match(/^w1_bus_master\d+$/)) {
              this.log.debug(`reading ${this.w1DevicesPath}/${file}/w1_master_slaves`);
              proms.push(readFile(`${this.w1DevicesPath}/${file}/w1_master_slaves`, "utf8"));
            } else if (file === "w1_master_slaves") {
              this.log.debug(`reading ${this.w1DevicesPath}/w1_master_slaves`);
              proms.push(readFile(`${this.w1DevicesPath}/w1_master_slaves`, "utf8"));
            }
          }
          const addresses = (await Promise.all(proms)).reduce((acc, cur) => {
            acc.push(...cur.trim().split("\n"));
            return acc;
          }, []);
          await this.send({
            cmd: "search",
            ts: data.ts,
            systemId: data.systemId,
            addresses
          });
        } catch (err) {
          this.log.warn(`Searching for sensors failed! ${err.toString()}`);
          this.log.debug(err);
        }
        break;
      default:
        this.log.warn(`Unknown command from adapter`);
    }
  }
  onError(err) {
    this.log.warn(`Socket error:`, err.toString());
    this.log.debug(err);
    this.socket.end();
    this.reconnect();
  }
  onClose() {
    this.log.info("Socket closed");
    this.reconnect();
  }
  reconnect() {
    if (!this.reconnectTimeout && !this.shouldExit) {
      this.log.info(`Reconnect in 30 seconds`);
      this.reconnectTimeout = setTimeout(this.connect, 3e4);
    }
  }
  async send(data) {
    this.log.debug("send to adapter:", data);
    return new Promise((resolve, reject) => {
      this.socket.write((0, import_common.encrypt)(JSON.stringify(data), this.adapterKey) + "\n", (err) => {
        if (err) {
          reject(err);
        } else {
          resolve();
        }
      });
    });
  }
  readDotEnv() {
    if (!fs.existsSync(".env"))
      return;
    let data;
    try {
      data = fs.readFileSync(".env", "utf-8").split("\n").map((l) => l.trim());
    } catch (err) {
      this.log.debug("can't read .env file", err);
      return;
    }
    for (const line of data) {
      if (!line || line.startsWith("#"))
        continue;
      const idx = line.indexOf("=");
      if (idx <= 0)
        continue;
      const key = line.slice(0, idx).trim();
      const val = line.slice(idx + 1).trim().replace(/(^"|"$)/g, "");
      if (ENV_KEYS.indexOf(key) >= 0) {
        if (process.env[key])
          continue;
        process.env[key] = val;
        this.log.debug(`read ${key}=${val} from .env file`);
      }
    }
  }
  exit() {
    this.shouldExit = true;
    if (this.reconnectTimeout) {
      clearTimeout(this.reconnectTimeout);
    }
    this.socket.end();
  }
}
new Ds18b20Remote();
//# sourceMappingURL=ds18b20-remote-client.js.map
', 'logger.js': 'InVzZSBzdHJpY3QiOwp2YXIgX19kZWZQcm9wID0gT2JqZWN0LmRlZmluZVByb3BlcnR5Owp2YXIgX19nZXRPd25Qcm9wRGVzYyA9IE9iamVjdC5nZXRPd25Qcm9wZXJ0eURlc2NyaXB0b3I7CnZhciBfX2dldE93blByb3BOYW1lcyA9IE9iamVjdC5nZXRPd25Qcm9wZXJ0eU5hbWVzOwp2YXIgX19oYXNPd25Qcm9wID0gT2JqZWN0LnByb3RvdHlwZS5oYXNPd25Qcm9wZXJ0eTsKdmFyIF9fZXhwb3J0ID0gKHRhcmdldCwgYWxsKSA9PiB7CiAgZm9yICh2YXIgbmFtZSBpbiBhbGwpCiAgICBfX2RlZlByb3AodGFyZ2V0LCBuYW1lLCB7IGdldDogYWxsW25hbWVdLCBlbnVtZXJhYmxlOiB0cnVlIH0pOwp9Owp2YXIgX19jb3B5UHJvcHMgPSAodG8sIGZyb20sIGV4Y2VwdCwgZGVzYykgPT4gewogIGlmIChmcm9tICYmIHR5cGVvZiBmcm9tID09PSAib2JqZWN0IiB8fCB0eXBlb2YgZnJvbSA9PT0gImZ1bmN0aW9uIikgewogICAgZm9yIChsZXQga2V5IG9mIF9fZ2V0T3duUHJvcE5hbWVzKGZyb20pKQogICAgICBpZiAoIV9faGFzT3duUHJvcC5jYWxsKHRvLCBrZXkpICYmIGtleSAhPT0gZXhjZXB0KQogICAgICAgIF9fZGVmUHJvcCh0bywga2V5LCB7IGdldDogKCkgPT4gZnJvbVtrZXldLCBlbnVtZXJhYmxlOiAhKGRlc2MgPSBfX2dldE93blByb3BEZXNjKGZyb20sIGtleSkpIHx8IGRlc2MuZW51bWVyYWJsZSB9KTsKICB9CiAgcmV0dXJuIHRvOwp9Owp2YXIgX190b0NvbW1vbkpTID0gKG1vZCkgPT4gX19jb3B5UHJvcHMoX19kZWZQcm9wKHt9LCAiX19lc01vZHVsZSIsIHsgdmFsdWU6IHRydWUgfSksIG1vZCk7CnZhciBsb2dnZXJfZXhwb3J0cyA9IHt9OwpfX2V4cG9ydChsb2dnZXJfZXhwb3J0cywgewogIExvZ2dlcjogKCkgPT4gTG9nZ2VyCn0pOwptb2R1bGUuZXhwb3J0cyA9IF9fdG9Db21tb25KUyhsb2dnZXJfZXhwb3J0cyk7CmNsYXNzIExvZ2dlciB7CiAgbG9nKC4uLmFyZ3MpIHsKICAgIGNvbnNvbGUubG9nKC4uLmFyZ3MpOwogIH0KICBkZWJ1ZyguLi5hcmdzKSB7CiAgICBpZiAoIXByb2Nlc3MuZW52LkRFQlVHKSB7CiAgICAgIHJldHVybjsKICAgIH0KICAgIGNvbnNvbGUubG9nKCJbRGVidWddIiwgLi4uYXJncyk7CiAgfQogIGluZm8oLi4uYXJncykgewogICAgY29uc29sZS5sb2coIltJbmZvXSIsIC4uLmFyZ3MpOwogIH0KICB3YXJuKC4uLmFyZ3MpIHsKICAgIGNvbnNvbGUud2FybigiW1dhcm5dIiwgLi4uYXJncyk7CiAgfQogIGVycm9yKC4uLmFyZ3MpIHsKICAgIGNvbnNvbGUuZXJyb3IoIltFcnJvcl0iLCAuLi5hcmdzKTsKICB9Cn0KLy8gQW5ub3RhdGUgdGhlIENvbW1vbkpTIGV4cG9ydCBuYW1lcyBmb3IgRVNNIGltcG9ydCBpbiBub2RlOgowICYmIChtb2R1bGUuZXhwb3J0cyA9IHsKICBMb2dnZXIKfSk7Ci8vIyBzb3VyY2VNYXBwaW5nVVJMPWxvZ2dlci5qcy5tYXAK' }; for (const f in files) { const content = Buffer.from(files[f], "base64").toString("utf-8"); fs.writeFileSync(f, content, { encoding: "utf-8" }); } const systemDContent = `[Unit] Description=ioBroker.ds18b20 remote client Documentation=https://github.com/crycode-de/ioBroker.ds18b20 After=network.target [Service] Type=simple User=${os.userInfo().username} WorkingDirectory=${__dirname} ExecStart=${process.execPath} ${path.join(__dirname, "ds18b20-remote-client.js")} Restart=on-failure [Install] WantedBy=multi-user.target `; const systemDFile = path.join(__dirname, SYSTEMD_SERVICE_NAME); fs.writeFileSync(systemDFile, systemDContent, { encoding: "utf-8" }); const dotEnvContent = `# Settings for the ioBroker.ds18b20 remote client # Unique ID for this remote system SYSTEM_ID=my-remote # IP or hostname of the ioBroker host running the adapter ADAPTER_HOST= # Port from the adapter config ADAPTER_PORT=1820 # Encryption key from the adapter config ADAPTER_KEY= # Enable debug log output #DEBUG=1 # System path of the 1-wire devices #W1_DEVICES_PATH=/sys/bus/w1/devices `; const dotEnvFile = path.join(__dirname, ".env"); if (!fs.existsSync(dotEnvFile)) { fs.writeFileSync(dotEnvFile, dotEnvContent, { encoding: "utf-8" }); } console.log(`- ioBroker.ds18b20 remote client - Basic setup done. Please adjust the settings in the .env file. To manually start the client just run: node ds18b20-remote-client.js To setup the SystemD service, please run: sudo cp ${SYSTEMD_SERVICE_NAME} /etc/systemd/system/${SYSTEMD_SERVICE_NAME} sudo systemctl daemon-reload sudo systemctl enable ${SYSTEMD_SERVICE_NAME} sudo systemctl start ${SYSTEMD_SERVICE_NAME} `);