Skip to content

Commit f08f756

Browse files
committed
lib: add navigator.onLine
1 parent 6dadb99 commit f08f756

File tree

5 files changed

+66
-0
lines changed

5 files changed

+66
-0
lines changed

doc/api/globals.md

+26
Original file line numberDiff line numberDiff line change
@@ -629,6 +629,32 @@ logical processors available to the current Node.js instance.
629629
console.log(`This process is running on ${navigator.hardwareConcurrency}`);
630630
```
631631

632+
### `navigator.onLine`
633+
634+
<!-- YAML
635+
added: REPLACEME
636+
-->
637+
638+
* {boolean}
639+
640+
The [`navigator.onLine`](https://html.spec.whatwg.org/multipage/system-state.html#dom-navigator-online-dev)
641+
read-only property returns the online status available
642+
of the current Node.js instance. If the Node.js instance is not able to detect
643+
to a local area network (LAN) or a router, it is offline; all other conditions
644+
return true. So while you can assume that the Node.js instance is offline when
645+
it returns a false value, you cannot assume that a true value necessarily
646+
means that the Node.js instance can access the internet. You could be getting
647+
false positives, such as in cases where the computer is running a
648+
virtualization software that has virtual ethernet adapters that are always
649+
"connected."
650+
651+
Therefore, if you really want to determine the online status of the Node.js
652+
instance, you should develop additional means for checking.
653+
654+
```js
655+
console.log(`This process is ${navigator.onLine ? 'online' : 'offline'}`);
656+
```
657+
632658
## `PerformanceEntry`
633659

634660
<!-- YAML

lib/internal/navigator.js

+9
Original file line numberDiff line numberDiff line change
@@ -15,6 +15,7 @@ const {
1515

1616
const {
1717
getAvailableParallelism,
18+
getOnLineStatus,
1819
} = internalBinding('os');
1920

2021
const kInitialize = Symbol('kInitialize');
@@ -37,10 +38,18 @@ class Navigator {
3738
this.#availableParallelism ??= getAvailableParallelism();
3839
return this.#availableParallelism;
3940
}
41+
42+
/**
43+
* @return {boolean}
44+
*/
45+
get onLine() {
46+
return getOnLineStatus();
47+
}
4048
}
4149

4250
ObjectDefineProperties(Navigator.prototype, {
4351
hardwareConcurrency: kEnumerableProperty,
52+
onLine: kEnumerableProperty,
4453
});
4554

4655
module.exports = {

src/node_os.cc

+28
Original file line numberDiff line numberDiff line change
@@ -247,6 +247,32 @@ static void GetInterfaceAddresses(const FunctionCallbackInfo<Value>& args) {
247247
args.GetReturnValue().Set(Array::New(isolate, result.data(), result.size()));
248248
}
249249

250+
static void GetOnLineStatus(const FunctionCallbackInfo<Value>& args) {
251+
Environment* env = Environment::GetCurrent(args);
252+
uv_interface_address_t* interfaces;
253+
int count, i;
254+
255+
int err = uv_interface_addresses(&interfaces, &count);
256+
257+
if (err == UV_ENOSYS) return args.GetReturnValue().Set(false);
258+
259+
if (err) {
260+
CHECK_GE(args.Length(), 1);
261+
env->CollectUVExceptionInfo(
262+
args[args.Length() - 1], errno, "uv_interface_addresses");
263+
return args.GetReturnValue().Set(false);
264+
}
265+
266+
for (i = 0; i < count; i++) {
267+
if (interfaces[i].is_internal == false) {
268+
uv_free_interface_addresses(interfaces, count);
269+
return args.GetReturnValue().Set(true);
270+
}
271+
}
272+
273+
uv_free_interface_addresses(interfaces, count);
274+
return args.GetReturnValue().Set(false);
275+
}
250276

251277
static void GetHomeDirectory(const FunctionCallbackInfo<Value>& args) {
252278
Environment* env = Environment::GetCurrent(args);
@@ -403,6 +429,7 @@ void Initialize(Local<Object> target,
403429
SetMethod(context, target, "getPriority", GetPriority);
404430
SetMethod(
405431
context, target, "getAvailableParallelism", GetAvailableParallelism);
432+
SetMethod(context, target, "getOnLineStatus", GetOnLineStatus);
406433
SetMethod(context, target, "getOSInformation", GetOSInformation);
407434
target
408435
->Set(context,
@@ -419,6 +446,7 @@ void RegisterExternalReferences(ExternalReferenceRegistry* registry) {
419446
registry->Register(GetFreeMemory);
420447
registry->Register(GetCPUInfo);
421448
registry->Register(GetInterfaceAddresses);
449+
registry->Register(GetOnLineStatus);
422450
registry->Register(GetHomeDirectory);
423451
registry->Register(GetUserInfo);
424452
registry->Register(SetPriority);

test/parallel/test-navigator.js

+2
Original file line numberDiff line numberDiff line change
@@ -13,3 +13,5 @@ const is = {
1313
is.number(+navigator.hardwareConcurrency, 'hardwareConcurrency');
1414
is.number(navigator.hardwareConcurrency, 'hardwareConcurrency');
1515
assert.ok(navigator.hardwareConcurrency > 0);
16+
17+
assert.strictEqual(typeof navigator.onLine, 'boolean');

typings/internalBinding/os.d.ts

+1
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,7 @@ export interface OSBinding {
1010
getFreeMem(): number;
1111
getCPUs(): Array<string | number>;
1212
getInterfaceAddresses(ctx: InternalOSBinding.OSContext): Array<string | number | boolean> | undefined;
13+
getOnLineStatus(): boolean;
1314
getHomeDirectory(ctx: InternalOSBinding.OSContext): string | undefined;
1415
getUserInfo(options: { encoding?: string } | undefined, ctx: InternalOSBinding.OSContext): {
1516
uid: number;

0 commit comments

Comments
 (0)