mirror of
				https://github.com/node-red/node-red-nodes.git
				synced 2025-03-01 10:37:43 +00:00 
			
		
		
		
	Add IPv6 support to node (#758)
This commit is contained in:
		| @@ -4,6 +4,14 @@ | ||||
|         <label for="node-input-host"><i class="fa fa-dot-circle-o"></i> <span data-i18n="ping.label.target"></span></label> | ||||
|         <input type="text" id="node-input-host" placeholder="192.168.0.1, www.google.com"> | ||||
|     </div> | ||||
|     <div class="form-row"> | ||||
|         <label for="node-protocol"><i class="fa fa-gear"></i> <span data-i18n="ping.label.protocol"></label> | ||||
|         <select type="text" id="node-input-protocol" style="width: 70%"> | ||||
|             <option value="Automatic" data-i18n="ping.label.protocol_option.auto"></option> | ||||
|             <option value="IPv4" data-i18n="ping.label.protocol_option.ipv4"></option> | ||||
|             <option value="IPv6" data-i18n="ping.label.protocol_option.ipv6"></option> | ||||
|         </select> | ||||
|     </div> | ||||
|     <div class="form-row"> | ||||
|         <label for="node-input-mode"><i class="fa fa-wrench"></i> <span data-i18n="ping.label.mode"></label> | ||||
|         <select type="text" id="node-input-mode" style="width: 70%"> | ||||
| @@ -54,6 +62,7 @@ var timerParameterValidator = function(node,v){ | ||||
|         category: "network-input", | ||||
|         color:"#fdf0c2", | ||||
|         defaults: { | ||||
|             protocol: {value:"Automatic"}, | ||||
|             mode: {value:"timed"}, | ||||
|             name: {value:""}, | ||||
|             host: {value:"", validate: function(v){ | ||||
|   | ||||
| @@ -6,7 +6,7 @@ module.exports = function(RED) { | ||||
|  | ||||
|     function doPing(node, host, arrayMode) { | ||||
|         const defTimeout = 5000; | ||||
|         var ex, hostOptions, commandLineOptions; | ||||
|         var ex, ex6, hostOptions, commandLineOptions; | ||||
|         if (typeof host === "string") { | ||||
|             hostOptions = { | ||||
|                 host: host, | ||||
| @@ -25,12 +25,33 @@ module.exports = function(RED) { | ||||
|         if (arrayMode) { | ||||
|             msg.ping = hostOptions | ||||
|         } | ||||
|  | ||||
|         //User Selected Protocol | ||||
|         if (plat == "linux" || plat == "android") { | ||||
|             commandLineOptions = ["-n", "-w", timeoutS, "-c", "1"] | ||||
|             if (node.protocol === "IPv4") { | ||||
|                 commandLineOptions = ["-n", "-4", "-w", timeoutS, "-c", "1"]; //IPv4 | ||||
|             } else if (node.protocol === "IPv6") { | ||||
|                 commandLineOptions = ["-n", "-6", "-w", timeoutS, "-c", "1"]; //IPv6 | ||||
|             } else { | ||||
|               commandLineOptions = ["-n", "-w", timeoutS, "-c", "1"]; //Automatic | ||||
|             } | ||||
|         } else if (plat.match(/^win/)) { | ||||
|             commandLineOptions = ["-n", "1", "-w", hostOptions.timeout] | ||||
|             if (node.protocol === "IPv4") { | ||||
|                 commandLineOptions = ["-n", "1", "-4", "-w", hostOptions.timeout]; //IPv4 | ||||
|             } else if (node.protocol === "IPv6") { | ||||
|                 commandLineOptions = ["-n", "1", "-6", "-w", hostOptions.timeout]; //IPv6 | ||||
|             } else { | ||||
|                 commandLineOptions = ["-n", "1", "-w", hostOptions.timeout]; //Automatic | ||||
|             } | ||||
|         } else if (plat == "darwin" || plat == "freebsd") { | ||||
|             commandLineOptions = ["-n", "-t", timeoutS, "-c", "1"] | ||||
|             if (node.protocol === "IPv4") { | ||||
|                 commandLineOptions = ["-n", "-4", "-t", timeoutS, "-c", "1"]; //IPv4 | ||||
|             } else if (node.protocol === "IPv6") { | ||||
|                 commandLineOptions = ["-n", "-6", "-t", timeoutS, "-c", "1"]; //IPv6 | ||||
|             } else { | ||||
|                 commandLineOptions = ["-n", "-t", timeoutS, "-c", "1"]; //Automatic | ||||
|             } | ||||
|  | ||||
|         } else { | ||||
|             node.error("Sorry - your platform - "+plat+" - is not recognised.", msg); | ||||
|             return; //dont pass go - just return! | ||||
| @@ -53,6 +74,23 @@ module.exports = function(RED) { | ||||
|         var fail = false; | ||||
|         //var regex = /from.*time.(.*)ms/; | ||||
|         var regex = /=.*[<|=]([0-9]*).*TTL|ttl..*=([0-9\.]*)/; | ||||
|  | ||||
|         var tryPing6 = false; | ||||
|         //catch error msg from ping | ||||
|         ex.stderr.setEncoding('utf8'); | ||||
|         ex.stderr.on("data", function (data) { | ||||
|             if (!data.includes('Usage')) { // !data: only get the error and not how to use the ping command | ||||
|                 if (data.includes('invalid') && data.includes('6')) { //if IPv6 not supported in version of ping try ping6 | ||||
|                     tryPing6 = true; | ||||
|                     //node.error(data, msg); // used for testing output of -6 on ping command. Keep this line untill out of beta testing please.  Contact---> https://discourse.nodered.org/u/meeki007 | ||||
|                 } else if (data.includes('Network is unreachable')) { | ||||
|                     node.error(data + "  Please check that your service provider or network device has IPv6 enabled", msg); | ||||
|                 } else { | ||||
|                     node.error(data, msg); | ||||
|                 } | ||||
|             } | ||||
|         }); | ||||
|  | ||||
|         ex.stdout.on("data", function (data) { | ||||
|             line += data.toString(); | ||||
|         }); | ||||
| @@ -72,20 +110,88 @@ module.exports = function(RED) { | ||||
|             } | ||||
|         }); | ||||
|         ex.on("close", function (code) { | ||||
|             if (fail) { fail = false; return; } | ||||
|             var m = regex.exec(line)||""; | ||||
|             if (m !== "") { | ||||
|                 if (m[1]) { res = Number(m[1]); } | ||||
|                 if (m[2]) { res = Number(m[2]); } | ||||
|             if (tryPing6 === false) { | ||||
|                 if (fail) { fail = false; return; } | ||||
|                 var m = regex.exec(line)||""; | ||||
|                 if (m !== "") { | ||||
|                     if (m[1]) { res = Number(m[1]); } | ||||
|                     if (m[2]) { res = Number(m[2]); } | ||||
|                 } | ||||
|                 if (code === 0) { msg.payload = res } | ||||
|                 try { node.send(msg); } | ||||
|                 catch(e) {console.warn(e)} | ||||
|  | ||||
|             } else { | ||||
|                 //fallback to ping6 for OS's that have not updated/out of date | ||||
|                 if (plat == "linux" || plat == "android") { | ||||
|                     commandLineOptions = ["-n", "-w", timeoutS, "-c", "1"]; | ||||
|                 } else if (plat == "darwin" || plat == "freebsd") { | ||||
|                     commandLineOptions = ["-n", "-c", "1"] //NOTE: -X / timeout does not work on mac OSX and most freebsd systems | ||||
|                 } else { | ||||
|                     node.error("Sorry IPv6 on your platform - "+plat+" - is not supported.", msg); | ||||
|                 } | ||||
|                 //spawn with timeout in case of os issue | ||||
|                 ex6 = spawn("ping6", [...commandLineOptions, hostOptions.host]); | ||||
|  | ||||
|                 //monitor every spawned process & SIGINT if too long | ||||
|                 var spawnTout = setTimeout(() => { | ||||
|                     node.log(`ping6 - Host '${hostOptions.host}' process timeout - sending SIGINT`) | ||||
|                     try { | ||||
|                         if (ex6 && ex6.pid) { ex6.kill("SIGINT"); } | ||||
|                     } | ||||
|                     catch(e) {console.warn(e); } | ||||
|                 }, hostOptions.timeout+1000); //add 1s for grace | ||||
|  | ||||
|                 //catch error msg from ping6 | ||||
|                 ex6.stderr.setEncoding('utf8'); | ||||
|                 ex6.stderr.on("data", function (data) { | ||||
|                     if (!data.includes('Usage')) { // !data: only get the error and not how to use the ping6 command | ||||
|                         if (data.includes('Network is unreachable')) { | ||||
|                               node.error(data + "  Please check that your service provider or network device has IPv6 enabled", msg); | ||||
|                           } else { | ||||
|                               node.error(data, msg); | ||||
|                         } | ||||
|                     } | ||||
|                 }); | ||||
|  | ||||
|                 ex6.stdout.on("data", function (data) { | ||||
|                     line += data.toString(); | ||||
|                 }); | ||||
|                 ex6.on("exit", function (err) { | ||||
|                     clearTimeout(spawnTout); | ||||
|                 }); | ||||
|                 ex6.on("error", function (err) { | ||||
|                     fail = true; | ||||
|                     if (err.code === "ENOENT") { | ||||
|                         node.error(err.code + " ping6 command not found", msg); | ||||
|                     } | ||||
|                     else if (err.code === "EACCES") { | ||||
|                         node.error(err.code + " can't run ping6 command", msg); | ||||
|                     } | ||||
|                     else { | ||||
|                         node.error(err.code, msg); | ||||
|                     } | ||||
|                 }); | ||||
|                 ex6.on("close", function (code) { | ||||
|                     if (fail) { fail = false; return; } | ||||
|                     var m = regex.exec(line)||""; | ||||
|                     if (m !== "") { | ||||
|                         if (m[1]) { res = Number(m[1]); } | ||||
|                         if (m[2]) { res = Number(m[2]); } | ||||
|                     } | ||||
|                     if (code === 0) { msg.payload = res } | ||||
|                     try { node.send(msg); } | ||||
|                     catch(e) {console.warn(e)} | ||||
|                 }); | ||||
|             } | ||||
|             if (code === 0) { msg.payload = res } | ||||
|             try { node.send(msg); } | ||||
|             catch(e) {console.warn(e)} | ||||
|  | ||||
|         }); | ||||
|  | ||||
|     } | ||||
|  | ||||
|     function PingNode(n) { | ||||
|         RED.nodes.createNode(this,n); | ||||
|         this.protocol = n.protocol||'Automatic'; | ||||
|         this.mode = n.mode; | ||||
|         this.host = n.host; | ||||
|         this.timer = n.timer * 1000; | ||||
| @@ -132,4 +238,4 @@ module.exports = function(RED) { | ||||
|         }); | ||||
|     } | ||||
|     RED.nodes.registerType("ping",PingNode); | ||||
| } | ||||
| } | ||||
|   | ||||
| @@ -15,6 +15,19 @@ | ||||
|     <p>Returns <b>false</b> if no response received, or if the host is unresolveable.</p> | ||||
|     <p>Default ping is every 20 seconds but can be configured.</p> | ||||
|  | ||||
|     <h4>Protocol...</h4> | ||||
|     <ul> | ||||
|         <li><b>Automatic</b><br> | ||||
|             <P>Will use any Protocol, IPv4 or IPv6, to reach host; based on your operating system network settings</P> | ||||
|         </li> | ||||
|         <li><b>IPv4</b><br> | ||||
|             <P>Forces use of IPv4 to reach host. Will fail if no route availibe</P> | ||||
|         </li> | ||||
|         <li><b>IPv6</b><br> | ||||
|             <P>Forces use of IPv6 to reach host. Will fail if no route availibe</P> | ||||
|         </li> | ||||
|     </ul> | ||||
|  | ||||
|     <h4>Mode...</h4> | ||||
|     <ul> | ||||
|         <li><b>Timed</b><br> | ||||
|   | ||||
| @@ -9,6 +9,12 @@ | ||||
|                 "timed": "Timed", | ||||
|                 "triggered": "Triggered" | ||||
|             }, | ||||
|             "protocol": "Protocol", | ||||
|             "protocol_option": { | ||||
|                 "auto": "Automatic", | ||||
|                 "ipv4": "IPv4", | ||||
|                 "ipv6": "IPv6" | ||||
|             }, | ||||
|             "tip": "Note: Leave Target field blank to allow msg.payload to set hosts dynamically." | ||||
|         } | ||||
|     } | ||||
|   | ||||
| @@ -15,6 +15,19 @@ | ||||
|     <p>ホスト名が解決できなかったり、レスポンスが得られなければ、<b>false</b>を返します。</p> | ||||
|     <p>デフォルトでは20秒ごとにpingを送りますが、設定で変更できます。</p> | ||||
|  | ||||
|     <h4>Protocol...</h4> | ||||
|     <ul> | ||||
|         <li><b>Automatic</b><br> | ||||
|             <P>Will use any Protocol, IPv4 or IPv6, to reach host; based on your operating system network settings</P> | ||||
|         </li> | ||||
|         <li><b>IPv4</b><br> | ||||
|             <P>Forces use of IPv4 to reach host. Will fail if no route availibe</P> | ||||
|         </li> | ||||
|         <li><b>IPv6</b><br> | ||||
|             <P>Forces use of IPv6 to reach host. Will fail if no route availibe</P> | ||||
|         </li> | ||||
|     </ul> | ||||
|  | ||||
|     <h4>モード...</h4> | ||||
|     <ul> | ||||
|         <li><b>時間</b><br> | ||||
| @@ -25,7 +38,7 @@ | ||||
|         <li><b>トリガー</b><br> | ||||
|             <p><code>トリガー</code>モードでは、入力ワイヤを結線し、<code>msg</code>入力によってping処理を起動します。 | ||||
|     <p><code>ターゲット</code>フィールドに入力した値はホスト名もしくはIPアドレスとして使用します。 ターゲットはホスト名/IPアドレスをカンマで区切ったものです。例: <code>"192.168.0.1"</code> あるいは <code>"192.168.0.1, www.google.com"</code></p> | ||||
|      | ||||
|  | ||||
|             <p><code>ターゲット</code>を空にした場合、カンマ区切り文字列、もしくは、ホストの配列を<code>msg.payload</code>に指定します。 | ||||
|                 <ul> | ||||
|                     <li><code>文字列</code> - カンマ区切りのホスト名/IPアドレス 例: <code>"192.168.0.1"</code> あるいは <code>"192.168.0.1, www.google.com"</code> </li> | ||||
| @@ -33,12 +46,12 @@ | ||||
|                 <li>配列ペイロードの例: <pre>[ | ||||
|     "192.168.0.99", | ||||
|     { | ||||
|         "host":"192.168.0.1",  | ||||
|         "host":"192.168.0.1", | ||||
|         "name":"ルータ" | ||||
|     },   | ||||
|     }, | ||||
|     { | ||||
|         "host":"myapiserver.com",  | ||||
|         "name":"拡張API",  | ||||
|         "host":"myapiserver.com", | ||||
|         "name":"拡張API", | ||||
|         "timeout": 20000, | ||||
|         "support":"support@myapiserver.com" | ||||
|     } | ||||
|   | ||||
| @@ -9,6 +9,12 @@ | ||||
|                 "timed": "時間", | ||||
|                 "triggered": "トリガー" | ||||
|             }, | ||||
|             "protocol": "プロトコル", | ||||
|             "protocol_option": { | ||||
|                 "auto": "自動の", | ||||
|                 "ipv4": "IPv4", | ||||
|                 "ipv6": "IPv6" | ||||
|             }, | ||||
|             "tip": "注: msg.payloadでホスト名を動的に指定する場合は、ターゲットフィールドを空にします。" | ||||
|         } | ||||
|     } | ||||
|   | ||||
		Reference in New Issue
	
	Block a user