Security News
pnpm 10.0.0 Blocks Lifecycle Scripts by Default
pnpm 10 blocks lifecycle scripts by default to improve security, addressing supply chain attack risks but sparking debate over compatibility and workflow changes.
@matter/node
Advanced tools
The Node API offers a high-level interface for implementing and interacting with Matter nodes. A node in Matter is a top-level network-accessible resource.
[!NOTE]
Don't confuse Matter "nodes" with Node.js. You can use the Node API from Node.js but it it not specific to Node.js.
This document gives a high-level overview of key Matter concepts and how they map to the Matter.js API.
You can look through Matter.js examples to get a feel for how you might use the Node API in your application. Familiarity with a few concepts described here might be helpful as you browse the examples.
---
title: The Node API
---
classDiagram
direction LR
class ClientNode {
TBD
}
ClientNode --|> Node
class ServerNode {
async run()
start()
cancel()
}
ServerNode --|> Node
class Node {
type: RootEndpointType
lifecycle: NodeLifecycle
}
Node --|> Endpoint
class Endpoint {
type: EndpointType
id: string
number: EndpointNumber
lifecycle: EndpointLifecycle
parts: Parts
lifecycle: Lifecycle
act(actor: agent => ...)
}
class Parts {
«iterable»
get(id)
}
Endpoint --* Parts
Parts --o Endpoint
class Agent {
«behavior id»: Behavior
context: ActionContext
}
Endpoint --> Agent
Agent --* Behavior
Agent --* ActionContext
class Behavior {
state: Object
events: EventEmitter
«command name»()
}
class EndpointLifecycle {
isInstalled: boolean
isReady: boolean
isTreeReady: boolean
}
class NodeLifecycle {
isOnline: boolean
isCommissioned: boolean
}
class ActionContext {
transaction: Transaction
fabric?: FabricIndex
subject?: SubjectId
}
NodeLifecycle --|> EndpointLifecycle
Endpoint --* EndpointLifecycle
Node --* NodeLifecycle
Most nodes are physical devices. The Matter specifications use the term device informally to describe nodes associated with a real-world device such as a light switch, door lock or window covering.
A controller is a node that manages other nodes and commissions them into a fabric specific to the controller. In Matter, a fabric is a private namespace associated with an specific controller. Nodes may be associated with more than one controller and thus more than one fabric.
Common controllers include Apple HomePod, Amazon Echo and Google Nest Hub.
A bridge is a node that publishes non-Matter devices as Matter nodes. Bridges are often provided by third parties to publish nodes for devices that do not have native Matter support. You can use Matter.js to create a bridge.
In Matter.js you can create a device or bridge using Server Node.
Server Node is a server-side implementation of the Node interface. The Node interface is common to both servers (devices and bridges) and clients (controllers).
[!NOTE]
Matter.js supports controllers via CommissioningController but we have not yet implemented the higher-level ClientNode class.
Here is how you instantiate a node and bring it online:
// ../../../matter-node.js-examples/src/tutorial/example01.ts
import { ServerNode } from "@matter/main";
const node = await ServerNode.create();
await node.run();
This example starts and publishes your node but of course you must add additional functionality to make the Node useful.
[!NOTE]
The boilerplate import referencing "matter-node.js" above is necessary to load Node.js-specific extensions to Matter.js.
A Matter endpoint is an individually addressable element of a Matter node.
Matter defines the functionality supported by an endpoint using a unique numerical endpoint type. The endpoint type designates a set of clusters associated with the endpoint. A cluster is a functional interface to a set of related functionality such as "window covering" or "pump control".
Matter.js defines endpoint types using EndpointType. We provide predefined EndpointTypes all endpoint types for all device types defined by the Matter standard in endpoint/definitions/device.
A node consists of a RootEndpoint and one or more other endpoints specific to the node's function.
In Matter.js endpoints are instances of the Endpoint class.
You can add an endpoint to a node by supplying the endpoint's DeviceType to the node:
// ../../../matter-node.js-examples/src/tutorial/example02.ts
import { OnOffLightDevice } from "@matter/main/devices/OnOffLightDevice";
import { ServerNode } from "@matter/main";
const node = await ServerNode.create();
await node.add(OnOffLightDevice);
await node.run();
This publishes a node with a working "on/off" cluster. However it's still not useful until you customize the on/off cluster to control your actual device.
A Matter.js Endpoint defines functionality as a composition of Behavior classes. A behavior is a JavaScript object that adds functionality to an endpoint.
Matter.js includes predefined behaviors for all standard Matter clusters. Some of these behaviors are fully functional without any effort on your part. In other cases the behavior is only useful if modified for your application.
The following example extends OnOffServer. OnOffServer is a Matter.js Behavior implementing the Matter "on/off" cluster.
// ../../../matter-node.js-examples/src/tutorial/example03.ts
import { OnOffLightDevice, OnOffLightRequirements } from "@matter/main/devices/OnOffLightDevice";
import { ServerNode } from "@matter/main";
class MyOnOffServer extends OnOffLightRequirements.OnOffServer {
override on() {
console.log("Turning on");
}
override off() {
console.log("Turning off");
}
}
const node = await ServerNode.create();
await node.add(OnOffLightDevice.with(MyOnOffServer));
await node.run();
This illustrates how you use Matter.js to customize behaviors. In this case we override the functionality for two commands on the Matter cluster, "on" and "off". A command is an element of a Matter cluster that other nodes may invoke remotely. Matter.js maps commands to specific behavior methods.
[!NOTE]
For
MyOnOffServer
we extendOnOffLightRequirements.OnOffServer
rather than extendingOnOffServer
directly. This is because the Matter specification specifies other mandatory changes to the on/off cluster. Matter.js makes those changes for us in OnOffLightRequirements.
In addition to commands, Matter clusters may define two other types of elements, "attributes" and "events".
An attribute is a value associated with a cluster that another node may read or write. A node may also subscribe to an attribute to receive updates when the attributes value changes.
Matter.js exposes attribute values in a Behavior property called "state". behavior.state
contains a property for each attribute supported by a cluster.
An event is data value that another node may subscribe to. Events are not associated with fixed values like attributes but events themselves are semi-persistent.
Matter.js exposes events in a Behavior property called "events". Each event supported by a
cluster has a corresponding behavior.events
property that is a Matter.js Observable.
Additionally, behavior.events
has a property of the form attributeName$change
for each cluster attribute. You may
use this event to listen for attribute changes.
In this example we use events and state to customize OnOffServer:
// ../../../matter-node.js-examples/src/tutorial/example04.ts
import { OnOffLightDevice } from "@matter/main/devices/OnOffLightDevice";
import { ServerNode } from "@matter/main";
const node = await ServerNode.create();
const light = await node.add(OnOffLightDevice);
light.events.onOff.onOff$Change.on(newValue => {
console.log(`Light is ${newValue ? "on" : "off"}`);
});
await node.run();
This example demonstrates how to use events. You define the events in the behavior but the endpoint makes them
available as endpoing.events.clusterName.eventName
. You can use methods on
and off
to subscribe and unsubscribe
to events.
Similarly to events, you define attributes in a behavior but the endpoint makes them available as
endpoint.state.clusterName.attributeName
;
We do not cover the following in detail here but call them out because you may encounter them in the Matter.js API.
A Matter.js ActionContext captures details about the remote node or internal process interacting with the Matter.js API. All interactions with the Node API have an ActionContext either implicitly or explicitly. Matter.js uses information from the context to enforce access rights and implement transactionality.
In Matter.js, changes to state are transactional. This means your state will not be visible in other contexts until
the transaction commits. If there is an error in your code, the transaction will roll back and your changes will be
reverted. This is controlled by the transaction
property of ActionContext
, which is a
Transaction.
Agent is a Matter.js class that brings together an
ActionContext with an Endpoint. Conceptually an
agent is a standin for an endpoint that provides direct access to the endpoint's behaviors. You can access the
ActionContext as agent.context
and each behavior instance as agent.behaviorName
.
For most interactions with an endpoint Matter.js manages agents, contexts and transactions for you. However in some
circumstances you may benefit from invoking commands locally or modifying attributes across multiple clusters
atomically. You can use endpoint.act
to obtain an Agent for this type of advanced use case.
In the following contrived example we use act
to manually invoke the Matter "toggle" command and then print the
results:
// ../../../matter-node.js-examples/src/tutorial/example05.ts
import { OnOffLightDevice } from "@matter/main/devices/on-off-light-device";
import { ServerNode } from "@matter/main";
const node = await ServerNode.create();
const light = await node.add(OnOffLightDevice);
await node.start();
await light.act(async agent => {
await agent.onOff.toggle();
console.log("On/off state after first toggle", agent.onOff.state);
await agent.onOff.toggle();
console.log("On/off state after second toggle", agent.onOff.state);
});
await node.close();
[!NOTE]
start
starts a node and waits for it to go online.close
shuts the node down and releases its resources.
We offer annotated examples that illustrate more advanced uses of the Matter.js API including bridges and other advanced device types.
FAQs
API for building Matter nodes
The npm package @matter/node receives a total of 11,145 weekly downloads. As such, @matter/node popularity was classified as popular.
We found that @matter/node demonstrated a healthy version release cadence and project activity because the last version was released less than a year ago. It has 0 open source maintainers collaborating on the project.
Did you know?
Socket for GitHub automatically highlights issues in each pull request and monitors the health of all your open source dependencies. Discover the contents of your packages and block harmful activity before you install or update your dependencies.
Security News
pnpm 10 blocks lifecycle scripts by default to improve security, addressing supply chain attack risks but sparking debate over compatibility and workflow changes.
Product
Socket now supports uv.lock files to ensure consistent, secure dependency resolution for Python projects and enhance supply chain security.
Research
Security News
Socket researchers have discovered multiple malicious npm packages targeting Solana private keys, abusing Gmail to exfiltrate the data and drain Solana wallets.