ChirpStack R1XゲートウェイとSenseCAP S2101の統合
はじめに

このガイドでは、Raspberry Piを搭載したSeeed reComputer R11エッジコントローラーでChirpStackを使用した完全なLoRaWANゲートウェイソリューションのセットアップ方法を説明します。WM1302 LoRaコンセントレーターモジュールにより、R1Xデバイスは信頼性の高い長距離無線通信が可能な強力なゲートウェイとして機能します。Semtech Packet Forwarderを設定することで、LoRaデータをネットワーク層とアプリケーション層を管理するChirpStackにシームレスに送信できます。Dockerを使用してChirpStackサービスのインストールとデプロイメントを簡素化し、モジュラーでスケーラブルなセットアップを確保します。最後に、システムはMQTTと統合され、SenseCAP S2101センサーなどのLoRaデバイスから世界中のどこからでもアクセス可能なアプリケーションへの安全でリアルタイムなIoTデータストリーミングを可能にします。
必要なハードウェア
| reComputer R1X | WM1302 LoRaWANゲートウェイモジュール | SenseCAP S2101 |
|---|---|---|
![]() | ![]() | ![]() |
Dockerインストールガイド
1. システムパッケージの更新
sudo apt update
sudo apt upgrade
2. Dockerのインストール
curl -fsSL https://get.docker.com -o get-docker.sh
sudo sh get-docker.sh
3. ユーザーをDockerグループに追加
sudo usermod -aG docker ${USER}
4. システムの再起動
sudo reboot
5. インストールの確認
docker run hello-world
6. Docker Composeのインストール
sudo apt install docker-compose
Packet Forwarderの実行
WM1302 LoRaコンセントレーターは、LoRaモジュールとChirpStack間でデータを中継するためにSemtech Packet Forwarderが必要です。reComputer R11はLoRaモジュール用の事前構築されたセットアップガイドを提供しています。
インストール手順については、公式Seeed Wikiを参照してください: Seeed reComputer R11 LoRaモジュールガイド
インストール後、以下の手順に従ってPacket Forwarderを設定して実行します。
1. 設定の変更
LoRa地域に対応する設定ファイルを開きます。例えば、US915の場合:
nano global_conf.json.sx1250.US915
gateway_confセクションを更新してChirpStackサーバーを指すようにします:
"gateway_conf": {
"gateway_ID": "AA555A0000000000",
/* change with default server address/ports */
"server_address": "localhost",
"serv_port_up": 1700,
"serv_port_down": 1700
}
AA555A0000000000を実際のゲートウェイIDに置き換えてください。そのままにしておきます 購入したモジュールに応じて、LoRaWAN地域に適したJSONファイルを使用してください。
ファイルを保存して終了します:
- CTRL + Xを押し、
- 次にYを押し、
- 最後にEnterを押します。
2. Packet Forwarderの開始
更新された設定を使用してPacket Forwarderを実行します:
./lora_pkt_fwd -c global_conf.json.sx1250.US915
ゲートウェイの開始
docker Composeファイルをダウンロードするには、reComputerでこのページにアクセスしてダウンロードする必要があります。リンク
次に、yamlファイルの設定に応じて周波数帯域を変更します
chirpstack-gateway-bridge:
image: chirpstack/chirpstack-gateway-bridge:4
restart: unless-stopped
ports:
- "1700:1700/udp"
volumes:
- ./configuration/chirpstack-gateway-bridge:/etc/chirpstack-gateway-bridge
environment:
- INTEGRATION__MQTT__EVENT_TOPIC_TEMPLATE=us915_0/gateway/{{ .GatewayID }}/event/{{ .EventType }}
- INTEGRATION__MQTT__STATE_TOPIC_TEMPLATE=us915_0/gateway/{{ .GatewayID }}/state/{{ .StateType }}
- INTEGRATION__MQTT__COMMAND_TOPIC_TEMPLATE=us915_0/gateway/{{ .GatewayID }}/command/#
depends_on:
- mosquitto
ChirpStackをインストールした後、R11 LoRaゲートウェイを登録してデータ処理を開始できます。
ChirpStackサービスの開始
まだ実行されていない場合は、すべてのChirpStackサービスを起動します:
sudo docker-compose up -d
コンテナが実行されていることを確認します:
sudo docker ps
ChirpStack Web UIへのアクセス
- Webブラウザを開いて以下にアクセスします:
http://localhost:8080/
- デフォルトの認証情報でログインします:
Username: admin
Password: admin
ゲートウェイの追加
- ChirpStack UIで、Gateways → Create Gatewayに移動します

-
以下の詳細を入力します:
- Gateway ID:
AA555A0000000000(実際のゲートウェイIDに置き換えてください) - Name: ゲートウェイの説明的な名前を付けます
- Gateway ID:

-
Create Gatewayをクリックして登録します。
-
この後、ChirpStack UIでゲートウェイを表示できるようになります

デバイスプロファイルの追加
LoRaWANデバイス(例:SenseCAP S2101)をChirpStackに接続するには、まずデバイスプロファイルを作成する必要があります。
-
Device Profiles → Create Device Profileに移動します
-
以下の詳細を入力します:
- Name: デバイスプロファイルの説明的な名前を付けます
- Region: デバイスとゲートウェイに一致する地域/サブバンドを選択します(例:
US915)

-
Codecタブに移動します:
- JavaScript Functionsを選択します
- デバイス用のコーデックを貼り付けます
⚠️ コーデックはLoRaデバイス固有です。例えば、Seeed S201xを使用している場合は、以下のコードを使用できます。 異なるデバイスを使用している場合は、正しいコーデックについてメーカーに相談してください。
- Uplink/Downlink Codecセクションにコーデックをコピー&ペーストし、プロファイルを保存します。

.js
function decodeUplink(input) {
return Decode(input.fPort, input.bytes, input.variables);
}
function Decode(fPort, bytes, variables) {
var bytesString = bytes2HexString(bytes).toLocaleUpperCase();
var fport = parseInt(fPort);
var decoded = {
valid: true,
err: 0,
payload: bytesString,
messages: []
};
// CRC check
if (!crc16Check(bytesString)) {
decoded['valid'] = false;
decoded['err'] = -1; // "crc check fail."
return { data: decoded };
}
// Length Check
if ((bytesString.length / 2 - 2) % 7 !== 0) {
decoded['valid'] = false;
decoded['err'] = -2; // "length check fail."
return { data: decoded };
}
// Cache sensor id
var sensorEuiLowBytes;
var sensorEuiHighBytes;
// Handle each frame
var frameArray = divideBy7Bytes(bytesString);
for (var forFrame = 0; forFrame < frameArray.length; forFrame++) {
var frame = frameArray[forFrame];
var channel = strTo10SysNub(frame.substring(0, 2));
var dataID = strTo10SysNub(frame.substring(2, 6));
var dataValue = frame.substring(6, 14);
var realDataValue = isSpecialDataId(dataID) ? ttnDataSpecialFormat(dataID, dataValue) : ttnDataFormat(dataValue);
if (checkDataIdIsMeasureUpload(dataID)) {
decoded.messages.push({
type: 'report_telemetry',
measurementId: dataID,
measurementValue: realDataValue
});
} else if (isSpecialDataId(dataID) || dataID === 5 || dataID === 6) {
switch (dataID) {
case 0x00: // node version
var versionData = sensorAttrForVersion(realDataValue);
decoded.messages.push({
type: 'upload_version',
hardwareVersion: versionData.ver_hardware,
softwareVersion: versionData.ver_software
});
break;
case 1: // sensor version
break;
case 2: // sensor eui low
sensorEuiLowBytes = realDataValue;
break;
case 3: // sensor eui high
sensorEuiHighBytes = realDataValue;
break;
case 7: // battery + interval
decoded.messages.push({
type: 'upload_battery',
battery: realDataValue.power
}, {
type: 'upload_interval',
interval: parseInt(realDataValue.interval) * 60
});
break;
case 9:
decoded.messages.push({
type: 'model_info',
detectionType: realDataValue.detectionType,
modelId: realDataValue.modelId,
modelVer: realDataValue.modelVer
});
break;
case 0x120: // remove sensor
decoded.messages.push({
type: 'report_remove_sensor',
channel: 1
});
break;
default:
break;
}
} else {
decoded.messages.push({
type: 'unknown_message',
dataID: dataID,
dataValue: dataValue
});
}
}
if (sensorEuiHighBytes && sensorEuiLowBytes) {
decoded.messages.unshift({
type: 'upload_sensor_id',
channel: 1,
sensorId: (sensorEuiHighBytes + sensorEuiLowBytes).toUpperCase()
});
}
return { data: decoded };
}
// ---------- Utils ----------
function crc16Check(data) {
return true;
}
function bytes2HexString(arrBytes) {
var str = '';
for (var i = 0; i < arrBytes.length; i++) {
var num = arrBytes[i];
var tmp = (num < 0 ? (255 + num + 1) : num).toString(16);
if (tmp.length === 1) tmp = '0' + tmp;
str += tmp;
}
return str;
}
function divideBy7Bytes(str) {
var frameArray = [];
for (var i = 0; i < str.length - 4; i += 14) {
frameArray.push(str.substring(i, i + 14));
}
return frameArray;
}
function littleEndianTransform(data) {
var arr = [];
for (var i = 0; i < data.length; i += 2) {
arr.push(data.substring(i, i + 2));
}
return arr.reverse();
}
function strTo10SysNub(str) {
var arr = littleEndianTransform(str);
return parseInt(arr.join(''), 16);
}
function checkDataIdIsMeasureUpload(dataId) {
return parseInt(dataId) > 4096;
}
function isSpecialDataId(dataID) {
switch (dataID) {
case 0:
case 1:
case 2:
case 3:
case 4:
case 7:
case 9:
case 0x120:
return true;
default:
return false;
}
}
function ttnDataSpecialFormat(dataId, str) {
var strReverse = littleEndianTransform(str);
if (dataId === 2 || dataId === 3) {
return strReverse.join('');
}
var str2 = toBinary(strReverse);
var arr = [];
switch (dataId) {
case 0: case 1: // versions
for (var k = 0; k < str2.length; k += 16) {
var tmp = str2.substring(k, k + 16);
tmp = (parseInt(tmp.substring(0, 8), 2) || 0) + '.' + (parseInt(tmp.substring(8, 16), 2) || 0);
arr.push(tmp);
}
return arr.join(',');
case 4:
for (var i = 0; i < str2.length; i += 8) {
var item = parseInt(str2.substring(i, i + 8), 2);
arr.push(item < 10 ? '0' + item : item.toString());
}
return arr.join('');
case 7:
return {
interval: parseInt(str2.substr(0, 16), 2),
power: parseInt(str2.substr(-16, 16), 2)
};
case 9:
return {
detectionType: parseInt(str2.substring(0, 8), 2),
modelId: parseInt(str2.substring(8, 16), 2),
modelVer: parseInt(str2.substring(16, 24), 2)
};
}
}
function ttnDataFormat(str) {
var strReverse = littleEndianTransform(str);
var str2 = toBinary(strReverse);
if (str2[0] === '1') {
var arr = str2.split('').map(b => b === '1' ? 0 : 1);
var val = parseInt(arr.join(''), 2) + 1;
return parseFloat('-' + val / 1000);
}
return parseInt(str2, 2) / 1000;
}
function sensorAttrForVersion(dataValue) {
var arr = dataValue.split(',');
return { ver_hardware: arr[0], ver_software: arr[1] };
}
function toBinary(arr) {
return arr.map(item => {
var bin = parseInt(item, 16).toString(2).padStart(8, '0');
return bin;
}).join('');
}
デバイスの追加
Device Profile が作成されたら、ChirpStack に LoRaWAN デバイスを登録できます。
- Tenant → Application に移動し、Add Application をクリックします

- アプリケーションの Name を入力して保存します
- 新しく作成したアプリケーションを開き、Add Device をクリックします

-
以下の詳細を入力します:
- Device EUI: LoRa デバイスの EUI を貼り付けます(デバイスのデータシートまたは設定ソフトウェア(例:SenseCAP アプリケーション)で確認できます)
- Device Profile: 先ほど作成したデバイスプロファイルを選択します

- Application Key を入力し、Submit をクリックします

デバイスステータスの確認
LoRaWAN デバイスを追加した後、デバイスが正しく接続され、データを送信していることを確認できます。
-
アプリケーションに移動し、追加したデバイスを選択します
-
Events タブに移動します
- デバイスがネットワークに正常に参加すると、join packet が表示されます

- パケットをクリックして詳細情報を表示します

- 例えば、SenseCAP S2101 などのデバイスから報告される温度と湿度のデータを確認できます
MQTT 統合
ChirpStack は MQTT を使用して LoRaWAN デバイスからアプリケーションやダッシュボードにデータをストリーミングします。これらのメッセージをリアルタイムで監視できます。

-
PC を reComputer R11 ゲートウェイと同じネットワークに接続します
-
MQTT Explorer などの MQTT クライアントを使用してトピックを購読します
-
MQTT クライアントを設定します:
- Host: reComputer R11 の IP アドレス
- Port:
1883
-
接続すると、デバイスを表すトピックのツリーが表示されます。例:
application/c853ffcd-53f0-4de3-83b9-5467ff895f76/device/2cf7f1c043500402/event/up
- トピックを展開すると、SenseCAP S2101 などのデバイスの温度や湿度などのセンサーデータを含むアップリンクメッセージが表示されます

Node-RED 統合
MQTT ノードとカスタム関数を使用して、Node-RED で LoRaWAN デバイスデータを可視化できます。
-
Node-RED を開き、MQTT IN ノードをフローにドラッグします
-
MQTT ノードを設定します:
- Server: reComputer R11 の IP(例:
10.0.0.208) - Port:
1883 - Topic:
application/+/device/+/event/up
- Server: reComputer R11 の IP(例:


-
MQTT メッセージペイロードをデコードするために Function node を追加します
- 例えば、JSON オブジェクトから温度と湿度を抽出します
// Get the JSON payload
let data = msg.payload;
if (typeof data === "string") {
try {
data = JSON.parse(data);
} catch (e) {
node.error("Invalid JSON", msg);
return [null, null];
}
}
// Check if "object" and "messages" exist
if (!data.object || !Array.isArray(data.object.messages)) {
node.warn("No messages found in payload");
return [null, null];
}
// Find the two measurements
let tempMsg = null;
let humMsg = null;
data.object.messages.forEach(m => {
if (m.type === "report_telemetry") {
if (m.measurementId === 4097) {
tempMsg = { topic: "temperature", payload: m.measurementValue };
} else if (m.measurementId === 4098) {
humMsg = { topic: "humidity", payload: m.measurementValue };
}
}
});
// Return 2 outputs: [temperature, humidity]
return [tempMsg, humMsg];
-
Function ノードから2つの出力ノードを接続します。1つは温度用、もう1つは湿度用です
-
各出力を Gauge node または Node-RED の他の可視化ノードに接続して、センサーの読み取り値を表示します


技術サポートと製品ディスカッション
弊社製品をお選びいただき、ありがとうございます!弊社製品での体験を可能な限りスムーズにするため、さまざまなサポートを提供しています。さまざまな好みやニーズに対応するため、複数のコミュニケーションチャンネルを用意しています。


