Mô phỏng một mạng Thread bằng OpenThread trong Docker

1. Giới thiệu

26b7f4f6b3ea0700.png.

OpenThread do Google phát hành là một phương thức triển khai nguồn mở của giao thức mạng Thread (Luồng). Google Nest đã phát hành OpenThread để cung cấp rộng rãi công nghệ cho các sản phẩm Nest cho các nhà phát triển nhằm đẩy nhanh quá trình phát triển sản phẩm cho ngôi nhà thông minh.

Đặc tả luồng xác định một giao thức truyền dữ liệu từ thiết bị tới thiết bị không dây công suất thấp, bảo mật và đáng tin cậy dựa trên IPv6 cho các ứng dụng gia đình. OpenThread triển khai tất cả các lớp mạng Thread bao gồm IPv6, 6LoWPAN, IEEE 802.15.4 với bảo mật MAC, Thiết lập liên kết lưới và Định tuyến lưới.

Lớp học lập trình này sẽ hướng dẫn bạn mô phỏng Mạng luồng trên thiết bị được mô phỏng bằng Docker.

Kiến thức bạn sẽ học được

  • Cách thiết lập chuỗi công cụ bản dựng OpenThread
  • Cách mô phỏng Mạng luồng
  • Cách xác thực các nút Luồng
  • Cách quản lý mạng Thread bằng OpenThread Daemon

Bạn cần có

  • Docker
  • Kiến thức cơ bản về Linux, định tuyến mạng

2. Thiết lập Docker

Lớp học lập trình này được thiết kế để sử dụng Docker trên máy Linux, Mac OS X hoặc Windows. Bạn nên dùng Linux.

Cài đặt Docker

Cài đặt Docker trên hệ điều hành mà bạn chọn.

Kéo hình ảnh Docker

Sau khi cài đặt Docker, hãy mở cửa sổ dòng lệnh và kéo hình ảnh Docker openthread/environment. Hình ảnh này có OpenThread và OpenThread Daemon được tạo sẵn và sẵn sàng sử dụng cho Lớp học lập trình này.

$ docker pull openthread/environment:latest

Lưu ý rằng có thể mất vài phút để tải xuống hoàn toàn.

Trong cửa sổ dòng lệnh, hãy khởi động một vùng chứa Docker trong hình ảnh rồi kết nối với shell bash của hình ảnh đó:

$ docker run --name codelab_otsim_ctnr -it --rm \
   --sysctl net.ipv6.conf.all.disable_ipv6=0 \
   --cap-add=net_admin openthread/environment bash

Tùy chọn --rm sẽ xóa vùng chứa khi bạn thoát khỏi vùng chứa. Không sử dụng tùy chọn này nếu bạn không muốn xóa vùng chứa.

Lưu ý cờ là cần thiết đối với Lớp học lập trình này:

  • --sysctl net.ipv6.conf.all.disable_ipv6=0 — cho phép IPv6 trong vùng chứa
  • --cap-add=net_admin — bật tính năng NET_ADMIN cho phép bạn thực thi các hoạt động liên quan đến mạng, chẳng hạn như thêm các tuyến IP

Khi ở trong vùng chứa, bạn sẽ có lời nhắc tương tự như sau:

root@c0f3912a74ff:/#

Trong ví dụ trên, c0f3912a74ff là mã vùng chứa. Mã vùng chứa cho thực thể của vùng chứa Docker sẽ khác với mã vùng chứa trong lời nhắc của Lớp học lập trình này.

Sử dụng Docker

Lớp học lập trình này giả định rằng bạn biết kiến thức cơ bản về cách sử dụng Docker. Bạn nên ở lại vùng chứa Docker trong toàn bộ Lớp học lập trình.

3. Mô phỏng một mạng Thread

Ứng dụng mẫu mà bạn sẽ sử dụng cho Lớp học lập trình này minh hoạ ứng dụng OpenThread tối thiểu hiển thị cấu hình OpenThread và các giao diện quản lý thông qua giao diện dòng lệnh cơ bản (CLI).

Bài tập này sẽ hướng dẫn bạn thực hiện các bước tối thiểu cần thiết để ping một thiết bị luồng được mô phỏng từ một thiết bị luồng khác được mô phỏng.

Hình bên dưới mô tả cấu trúc liên kết mạng Thread cơ bản. Đối với bài tập này, chúng ta sẽ mô phỏng hai nút trong vòng tròn màu xanh lục: một Thread Leader và Thread Router có một kết nối duy nhất giữa chúng.

6e3aa07675f902dc.png.

Tạo mạng

1. Bắt đầu Nút 1

Nếu bạn chưa làm như vậy, trong cửa sổ dòng lệnh, hãy khởi động vùng chứa Docker và kết nối với shell bash:

$ docker run --name codelab_otsim_ctnr -it --rm \
   --sysctl net.ipv6.conf.all.disable_ipv6=0 \
   --cap-add=net_admin openthread/environment bash

Trong vùng chứa Docker, hãy tạo quy trình CLI cho thiết bị luồng được mô phỏng bằng cách sử dụng tệp nhị phân ot-cli-ftd.

root@c0f3912a74ff:/# /openthread/build/examples/apps/cli/ot-cli-ftd 1

Lưu ý: Nếu bạn không thấy lời nhắc > sau khi chạy lệnh này, hãy nhấn enter.

Tệp nhị phân này triển khai một thiết bị OpenThread. Trình điều khiển vô tuyến IEEE 802.15.4 được triển khai trên UDP (khung hình IEEE 802.15.4 được truyền trong tải trọng UDP).

Đối số của 1 là bộ mô tả tệp đại diện cho các bit ít quan trọng nhất của IEEE EUI-64 "được nhà máy chỉ định" cho thiết bị được mô phỏng. Giá trị này cũng được sử dụng khi liên kết với cổng UDP cho mô phỏng vô tuyến IEEE 802.15.4 (port = 9000 + bộ mô tả tệp). Mỗi thực thể của một thiết bị Luồng (EmThread) được mô phỏng trong Lớp học lập trình này sẽ sử dụng một bộ mô tả tệp khác.

Lưu ý: Chỉ sử dụng phần mô tả tệp 1 trở lên như đã nêu trong Lớp học lập trình này khi tạo quy trình cho thiết bị được mô phỏng. Chỉ số mô tả tệp là 0 được dành để sử dụng cho mục đích khác.

Tạo Tập dữ liệu hoạt động mới và cam kết tập dữ liệu đó là tập dữ liệu đang hoạt động. Tập dữ liệu hoạt động là cấu hình cho mạng Luồng mà bạn đang tạo.

> dataset init new
Done
> dataset
Active Timestamp: 1
Channel: 20
Channel Mask: 07fff800
Ext PAN ID: d6263b6d857647da
Mesh Local Prefix: fd61:2344:9a52:ede0/64
Network Key: e4344ca17d1dca2a33f064992f31f786
Network Name: OpenThread-c169
PAN ID: 0xc169
PSKc: ebb4f2f8a68026fc55bcf3d7be3e6fe4
Security Policy: 0, onrcb
Done

Xác nhận tập dữ liệu này là tập dữ liệu đang hoạt động:

> dataset commit active
Done

Hiển thị giao diện IPv6:

> ifconfig up
Done

Bắt đầu hoạt động của giao thức Thread:

> thread start
Done

Chờ vài giây và xác minh rằng thiết bị đã trở thành Trình quản lý chuỗi. Người lãnh đạo là thiết bị chịu trách nhiệm quản lý việc chỉ định ID bộ định tuyến.

> state
leader
Done

Xem các địa chỉ IPv6 được gán cho giao diện Chủ đề của Nút 1 (đầu ra của bạn sẽ khác):

> ipaddr
fd61:2344:9a52:ede0:0:ff:fe00:fc00
fd61:2344:9a52:ede0:0:ff:fe00:5000
fd61:2344:9a52:ede0:d041:c5ba:a7bc:5ce6
fe80:0:0:0:94da:92ea:1353:4f3b
Done

Lưu ý các loại địa chỉ IPv6 cụ thể:

  • Bắt đầu bằng fd = mesh-local
  • Bắt đầu bằng fe80 = link-local

Các loại địa chỉ lưới địa phương được phân loại thêm:

  • Chứa ff:fe00 = Bộ định tuyến bộ định tuyến (RLOC)
  • Không chứa ff:fe00 = Mã định danh điểm cuối (EID)

Xác định EID trong dữ liệu đầu ra của Play Console rồi ghi chú để sử dụng sau. Trong đầu ra mẫu ở trên, EID là:

fd61:2344:9a52:ede0:d041:c5ba:a7bc:5ce6

2. Bắt đầu Nút 2

Mở một thiết bị đầu cuối mới và thực thi một shell bash trong vùng chứa Docker hiện đang chạy để sử dụng cho Nút 2.

$ docker exec -it codelab_otsim_ctnr bash

Tại lời nhắc bash mới này, tạo ra quy trình CLI với đối số 2. Đây là thiết bị Luồng mô phỏng thứ hai của bạn:

root@c0f3912a74ff:/# /openthread/build/examples/apps/cli/ot-cli-ftd 2

Lưu ý: Nếu bạn không thấy lời nhắc > sau khi chạy lệnh này, hãy nhấn enter.

Định cấu hình Khóa mạng Thread và PAN ID, sử dụng cùng các giá trị như Tập dữ liệu hoạt động của Nút 1:

> dataset networkkey e4344ca17d1dca2a33f064992f31f786
Done
> dataset panid 0xc169
Done

Xác nhận tập dữ liệu này là tập dữ liệu đang hoạt động:

> dataset commit active
Done

Hiển thị giao diện IPv6:

> ifconfig up
Done

Bắt đầu hoạt động của giao thức Thread:

> thread start
Done

Thiết bị sẽ tự khởi chạy dưới dạng Cấp độ con. Thread Child tương đương với End Device (Thiết bị cuối) là thiết bị Thread (truyền) và chỉ nhận lưu lượng truy cập đơn (unicast) với thiết bị mẹ.

> state
child
Done

Trong vòng 2 phút, bạn sẽ thấy trạng thái chuyển đổi từ child sang router. Bộ định tuyến luồng có khả năng định tuyến lưu lượng truy cập giữa các thiết bị luồng. Đây còn được gọi là cấp độ gốc.

> state
router
Done

Xác minh mạng

Một cách dễ dàng để xác minh mạng lưới là xem bảng bộ định tuyến.

1. Kiểm tra kết nối

Trên Nút 2, nhận RLOC16. RLOC16 là 16 bit cuối cùng của địa chỉ RLOC IPv6 của thiết bị.

> rloc16
5800
Done

Trên Nút 1, hãy kiểm tra bảng bộ định tuyến cho RLOC16 của Nút 2. Trước tiên, hãy đảm bảo Nút 2 chuyển sang trạng thái bộ định tuyến.

> router table
| ID | RLOC16 | Next Hop | Path Cost | LQ In  | LQ Out  | Age | Extended MAC   |
+----+--------+----------+-----------+--------+-------+---+--------------------+
| 20 | 0x5000 |       63 |         0 |      0 |     0 |   0 | 96da92ea13534f3b |
| 22 | 0x5800 |       63 |         0 |      3 |     3 |  23 | 5a4eb647eb6bc66c |

Đã tìm thấy RLOC của 0x5800 của nút 2 trong bảng, xác nhận rằng nút này đã được kết nối với lưới.

2. Ping nút 1 từ nút 2

Xác minh khả năng kết nối giữa hai thiết bị luồng được mô phỏng. Trong Nút 2, ping EID được chỉ định cho Nút 1:

> ping fd61:2344:9a52:ede0:d041:c5ba:a7bc:5ce6
> 16 bytes from fd61:2344:9a52:ede0:d041:c5ba:a7bc:5ce6: icmp_seq=1 hlim=64 time=12ms

Nhấn enter để quay lại lời nhắc CLI >.

Kiểm tra mạng

Bây giờ, bạn có thể ping thành công giữa 2 thiết bị Thread được mô phỏng, hãy kiểm tra mạng lưới bằng cách thực hiện ngoại tuyến một nút.

Quay lại Nút 1 và dừng Luồng:

> thread stop
Done

Chuyển sang Nút 2 và kiểm tra trạng thái. Trong vòng hai phút, Node 2 phát hiện ra rằng biến thể dẫn đầu (Node 1) đang ngoại tuyến và bạn sẽ thấy Node 2 chuyển đổi thành leader của mạng:

> state
router
Done
...
> state
leader
Done

Sau khi được xác nhận, hãy dừng Thread và khôi phục cài đặt gốc cho Node 2 trước khi quay lại lời nhắc bash của Docker. Việc đặt lại về trạng thái ban đầu được thực hiện nhằm đảm bảo thông tin xác thực của luồng Thread mà chúng ta dùng trong bài tập này sẽ không được chuyển sang bài tập tiếp theo.

> thread stop
Done
> factoryreset
>
> exit
root@c0f3912a74ff:/#

Bạn có thể phải nhấn enter vài lần để đưa lại lời nhắc > sau lệnh factoryreset. Không thoát khỏi vùng chứa Docker.

Cũng khôi phục cài đặt gốc và thoát Nút 1:

> factoryreset
>
> exit
root@c0f3912a74ff:/#

Xem Tài liệu tham khảo CLI về OpenThread để khám phá tất cả các lệnh CLI hiện có.

4. Xác thực nút bằng cách Ủy quyền

Trong bài tập trước, bạn đã thiết lập mạng Thread với hai thiết bị mô phỏng và kết nối đã xác minh. Tuy nhiên, việc này chỉ cho phép lưu lượng truy cập cục bộ qua đường liên kết IPv6 chưa được xác thực truyền qua lại giữa các thiết bị. Để định tuyến lưu lượng truy cập IPv6 toàn cục giữa chúng (và Internet qua bộ định tuyến biên Thread), các nút phải được xác thực.

Để xác thực, một thiết bị phải có vai trò là Ủy viên. Ủy viên hội đồng là máy chủ xác thực hiện được chọn cho các thiết bị Chuỗi mới, đồng thời là người ủy quyền cung cấp thông tin đăng nhập mạng cần thiết để các thiết bị tham gia mạng này.

Trong bài tập này, chúng ta sẽ sử dụng cùng một cấu trúc liên kết hai nút như trước đây. Để xác thực, Người quản lý luồng sẽ đóng vai trò là Ủy viên, Bộ định tuyến luồng với tư cách là Người tham gia.

d6a67e8a0d0b5dcb.png.

Docker

Đối với mỗi Nút (cửa sổ đầu cuối) trong các bài tập còn lại, hãy đảm bảo bạn đang chạy vùng chứa Docker với bản dựng OpenThread. Nếu tiếp tục từ bài tập trước, bạn vẫn nên có hai lời nhắc bash trong cùng một vùng chứa Docker đã mở. Nếu không, hãy xem bước Khắc phục sự cố docker hoặc chỉ cần làm lại bài tập Mô phỏng mạng luồng.

1. Tạo mạng

Trong Nút 1, tạo ra quy trình CLI:

root@c0f3912a74ff:/# /openthread/build/examples/apps/cli/ot-cli-ftd 1

Lưu ý: Nếu bạn không thấy lời nhắc > sau khi chạy lệnh này, hãy nhấn enter.

Tạo Tập dữ liệu hoạt động mới, xác nhận tập dữ liệu này là tập dữ liệu đang hoạt động và bắt đầu Luồng:

> dataset init new
Done
> dataset
Active Timestamp: 1
Channel: 12
Channel Mask: 07fff800
Ext PAN ID: e68d05794bf13052
Mesh Local Prefix: fd7d:ddf7:877b:8756/64
Network Key: a77fe1d03b0e8028a4e13213de38080e
Network Name: OpenThread-8f37
PAN ID: 0x8f37
PSKc: f9debbc1532487984b17f92cd55b21fc
Security Policy: 0, onrcb
Done

Xác nhận tập dữ liệu này là tập dữ liệu đang hoạt động:

> dataset commit active
Done

Hiển thị giao diện IPv6:

> ifconfig up
Done

Bắt đầu hoạt động của giao thức Thread:

> thread start
Done

Chờ vài giây và xác minh rằng thiết bị đã trở thành Trình quản lý chuỗi:

> state
leader
Done

2. Bắt đầu vai trò Ủy viên

Trong khi vẫn ở Nút 1, hãy bắt đầu vai trò Ủy viên:

> commissioner start
Done

Cho phép bất kỳ Người tham gia nào (bằng cách sử dụng ký tự đại diện *) có Thông tin đăng nhập chung của J01NME để đưa vào mạng. Người liên kết là một thiết bị do quản trị viên con người thêm vào Mạng luồng được ủy quyền.

> commissioner joiner add * J01NME
Done

3. Bắt đầu vai trò Người tham gia

Trong cửa sổ dòng lệnh thứ hai, trong vùng chứa Docker, tạo ra một quy trình CLI mới. Đây là Nút 2.

root@c0f3912a74ff:/# /openthread/build/examples/apps/cli/ot-cli-ftd 2

Trên Nút 2, hãy bật vai trò Người kết hợp bằng cách sử dụng Thông tin xác thực người tham gia J01NME.

> ifconfig up
Done
> joiner start J01NME
Done

... đợi vài giây để xác nhận ...

Join success

Với tư cách là Người tham gia, thiết bị (Nút 2) đã tự xác thực thành công với Ủy viên (Nút 1) và nhận được thông tin đăng nhập Mạng luồng.

Bây giờ, Nút 2 đã được xác thực, hãy bắt đầu Luồng:

> thread start
Done

4. Xác thực mạng

Kiểm tra state trên Nút 2 để xác thực rằng nút đó hiện đã tham gia mạng. Trong vòng hai phút, Nút 2 chuyển đổi từ child sang router:

> state
child
Done
...
> state
router
Done

5. Đặt lại cấu hình

Để chuẩn bị cho bài tập tiếp theo, hãy đặt lại cấu hình. Trên mỗi Nút, dừng Luồng, thiết lập cài đặt gốc và thoát khỏi thiết bị Luồng được mô phỏng:

> thread stop
Done
> factoryreset
>
> exit
root@c0f3912a74ff:/#

Bạn có thể phải nhấn enter vài lần để đưa lại lời nhắc > sau lệnh factoryreset.

5. Quản lý mạng bằng OpenThread Daemon

Đối với bài tập này, chúng ta sẽ mô phỏng một thực thể CLI (một thiết bị SoC Thread được nhúng) và một thực thể Radio Co-Processor (RCP).

ot-daemon là một chế độ của ứng dụng OpenThread Posix sử dụng ổ cắm UNIX làm đầu vào và đầu ra, để lõi OpenThread có thể chạy dưới dạng một dịch vụ. Máy khách có thể giao tiếp với dịch vụ này bằng cách kết nối với ổ cắm có OpenThread CLI làm giao thức.

ot-ctl là CLI do ot-daemon cung cấp để quản lý và định cấu hình RCP. Bằng cách này, chúng ta sẽ kết nối RCP với mạng do thiết bị Thread tạo.

Docker

Đối với mỗi Nút (cửa sổ đầu cuối) trong bài tập này, hãy đảm bảo bạn đang chạy vùng chứa Docker với bản dựng OpenThread. Nếu tiếp tục từ bài tập trước, bạn nên có hai lời nhắc bash trong cùng một vùng chứa Docker đã mở. Nếu không, hãy xem bước Khắc phục sự cố docker.

Sử dụng ot-daemon

Bài tập này sẽ sử dụng 3 cửa sổ dòng lệnh, tương ứng với các cửa sổ sau:

  1. Bản sao CLI của thiết bị Chủ đề được mô phỏng (Nút 1)
  2. ot-daemon quá trình
  3. Phiên bản CLI ot-ctl

1. Bắt đầu Nút 1

Trong cửa sổ dòng lệnh đầu tiên, hãy tạo quy trình CLI cho thiết bị luồng được mô phỏng:

root@c0f3912a74ff:/# /openthread/build/examples/apps/cli/ot-cli-ftd 1

Lưu ý: Nếu bạn không thấy lời nhắc > sau khi chạy lệnh này, hãy nhấn enter.

Tạo Tập dữ liệu hoạt động mới, xác nhận tập dữ liệu này là tập dữ liệu đang hoạt động và bắt đầu Luồng:

> dataset init new
Done
> dataset
Active Timestamp: 1
Channel: 13
Channel Mask: 07fff800
Ext PAN ID: 97d584bcd493b824
Mesh Local Prefix: fd55:cf34:dea5:7994/64
Network Key: ba6e886c7af50598df1115fa07658a83
Network Name: OpenThread-34e4
PAN ID: 0x34e4
PSKc: 38d6fd32c866927a4dfcc06d79ae1192
Security Policy: 0, onrcb
Done

Xác nhận tập dữ liệu này là tập dữ liệu đang hoạt động:

> dataset commit active
Done

Hiển thị giao diện IPv6:

> ifconfig up
Done

Bắt đầu hoạt động của giao thức Thread:

> thread start
Done

Xem các địa chỉ IPv6 được gán cho giao diện luồng của nút 1:

> ipaddr
fd55:cf34:dea5:7994:0:ff:fe00:fc00
fd55:cf34:dea5:7994:0:ff:fe00:d000
fd55:cf34:dea5:7994:460:872c:e807:c4ab
fe80:0:0:0:9cd8:aab6:482f:4cdc
Done
>

Như đã giải thích trong bước Mô phỏng mạng chuỗi, một địa chỉ là liên kết cục bộ (fe80) và ba địa chỉ là cục bộ lưới (fd). EID là địa chỉ lưới cục bộ không chứa ff:fe00 trong địa chỉ. Trong mẫu đầu ra này, EID là fd55:cf34:dea5:7994:460:872c:e807:c4ab.

Xác định EID cụ thể từ đầu ra ipaddr sẽ được dùng để giao tiếp với nút này.

2. Bắt đầu ot-daemon

Trong cửa sổ dòng lệnh thứ hai, hãy tạo một nút thiết bị tun rồi đặt quyền đọc/ghi:

root@c0f3912a74ff:/# mkdir -p /dev/net && mknod /dev/net/tun c 10 200
root@c0f3912a74ff:/# chmod 600 /dev/net/tun

Thiết bị này dùng để truyền và nhận gói tin trong thiết bị ảo. Bạn có thể gặp lỗi nếu thiết bị đã được tạo. Điều này là bình thường và bạn có thể bỏ qua.

Khởi động ot-daemon cho một nút RCP mà chúng ta sẽ gọi là Nút 2. Sử dụng cờ chi tiết -v để bạn có thể xem đầu ra nhật ký và xác nhận rằng nó đang chạy:

root@c0f3912a74ff:/# /openthread/build/posix/src/posix/ot-daemon -v \
'spinel+hdlc+forkpty:///openthread/build/examples/apps/ncp/ot-rcp?forkpty-arg=2'

Khi thành công, ot-daemon ở chế độ chi tiết sẽ tạo đầu ra tương tự như sau:

ot-daemon[31]: Running OPENTHREAD/297a880; POSIX; Feb  1 2022 04:43:39
ot-daemon[31]: Thread version: 3
ot-daemon[31]: Thread interface: wpan0
ot-daemon[31]: RCP version: OPENTHREAD/297a880; SIMULATION; Feb  1 2022 04:42:50

Để thiết bị đầu cuối này ở trạng thái mở và chạy trong nền. Bạn sẽ không nhập thêm lệnh nào trong đó.

3. Sử dụng ot-ctl để kết nối mạng

Chúng tôi chưa uỷ quyền cho Nút 2 (RCP ot-daemon) cho bất kỳ mạng Thread nào. Đây là nơi ot-ctl xuất hiện. ot-ctl sử dụng CLI tương tự như ứng dụng CLI OpenThread. Do đó, bạn có thể kiểm soát các nút ot-daemon theo cách tương tự như các thiết bị Luồng đã mô phỏng khác.

Mở cửa sổ dòng lệnh thứ ba và thực thi vùng chứa hiện tại:

$ docker exec -it codelab_otsim_ctnr bash

Khi ở trong vùng chứa, hãy khởi động ot-ctl:

root@c0f3912a74ff:/# /openthread/build/posix/src/posix/ot-ctl
>

Bạn sẽ sử dụng ot-ctl trong cửa sổ dòng lệnh thứ ba này để quản lý Nút 2 (nút RCP) mà bạn đã bắt đầu trong cửa sổ dòng lệnh thứ hai với ot-daemon. Kiểm tra state của Nút 2:

> state
disabled
Done

Tải eui64 của Nút 2 để hạn chế tham gia vào Người tham gia cụ thể:

> eui64
18b4300000000001
Done

Trên Nút 1 (cửa sổ dòng lệnh đầu tiên), hãy khởi động Ủy viên và hạn chế chỉ tham gia eui64 đó:

> commissioner start
Done
> commissioner joiner add 18b4300000000001 J01NME
Done

Trong cửa sổ dòng lệnh thứ ba, hiển thị giao diện mạng cho Nút 2 và tham gia mạng:

> ifconfig up
Done
> joiner start J01NME
Done

... đợi vài giây để xác nhận ...

Join success

Với tư cách là Người tham gia, RCP (Nút 2) đã tự xác thực thành công với Ủy viên (Nút 1) và nhận được thông tin xác thực của Mạng Thread.

Bây giờ hãy kết nối Nút 2 với mạng Luồng (một lần nữa, trong cửa sổ dòng lệnh thứ ba):

> thread start
Done

4. Xác thực mạng

Trong thiết bị đầu cuối thứ ba, hãy kiểm tra state trên Node 2, để xác thực rằng thiết bị hiện đã tham gia mạng. Trong vòng hai phút, Nút 2 chuyển đổi từ child sang router:

> state
child
Done
...
> state
router
Done

5. Xác thực kết nối

Trong cửa sổ dòng lệnh thứ ba, hãy thoát ot-ctl bằng cách sử dụng lệnh Ctrl+D hoặc exit rồi quay lại bảng điều khiển bash của vùng chứa. Từ bảng điều khiển này, ping Node 1, dùng EID của nút đó với lệnh ping6. Nếu thực thể RCP ot-daemon được liên kết và giao tiếp thành công với mạng Thread, thì quá trình ping thành công:

root@c0f3912a74ff:/# ping6 -c 4 fd55:cf34:dea5:7994:460:872c:e807:c4ab
PING fd55:cf34:dea5:7994:460:872c:e807:c4ab (fd55:cf34:dea5:7994:460:872c:e807:c4ab): 56 data bytes
64 bytes from fd55:cf34:dea5:7994:460:872c:e807:c4ab: icmp_seq=0 ttl=64 time=4.568 ms
64 bytes from fd55:cf34:dea5:7994:460:872c:e807:c4ab: icmp_seq=1 ttl=64 time=6.396 ms
64 bytes from fd55:cf34:dea5:7994:460:872c:e807:c4ab: icmp_seq=2 ttl=64 time=7.594 ms
64 bytes from fd55:cf34:dea5:7994:460:872c:e807:c4ab: icmp_seq=3 ttl=64 time=5.461 ms
--- fd55:cf34:dea5:7994:460:872c:e807:c4ab ping statistics ---
4 packets transmitted, 4 packets received, 0% packet loss
round-trip min/avg/max/stddev = 4.568/6.005/7.594/1.122 ms

6. Khắc phục sự cố của Docker

Nếu bạn đã thoát khỏi vùng chứa Docker

bash lời nhắc, bạn có thể cần phải kiểm tra xem quảng cáo có đang chạy hay không và khởi động lại / vào lại nếu cần. Mọi vùng chứa Docker mà bạn đã tạo trong đó bạn không sử dụng tuỳ chọn --rm sẽ vẫn tồn tại.

Cách cho biết vùng chứa Docker nào đang chạy:

$ docker ps
CONTAINER ID        IMAGE               COMMAND             CREATED             STATUS              PORTS               NAMES
505fc57ffc72        environment       "bash"              10 minutes ago      Up 10 minutes                           codelab_otsim_ctnr

Để hiển thị tất cả các vùng chứa Docker (cả đang chạy và đã dừng):

$ docker ps -a
CONTAINER ID        IMAGE               COMMAND             CREATED             STATUS              PORTS               NAMES
505fc57ffc72        environment       "bash"              10 minutes ago      Up 10 minutes                           codelab_otsim_ctnr

Nếu bạn không thấy vùng chứa codelab_otsim_ctnr trong đầu ra của một trong hai lệnh docker ps, hãy chạy lại bộ chứa đó:

$ docker run --name codelab_otsim_ctnr -it --rm \
   --sysctl net.ipv6.conf.all.disable_ipv6=0 \
   --cap-add=net_admin openthread/environment bash

Chỉ sử dụng tùy chọn --rm nếu bạn muốn xóa vùng chứa khi thoát khỏi vùng chứa.

Nếu vùng chứa đã dừng (được liệt kê trong docker ps -a nhưng không có trong docker ps), hãy khởi động lại vùng chứa:

$ docker start -i codelab_otsim_ctnr

Nếu vùng chứa Docker đang chạy (liệt kê trong docker ps), hãy kết nối lại với vùng chứa trong từng thiết bị đầu cuối:

$ docker exec -it codelab_otsim_ctnr bash

Lỗi "Hoạt động không được phép"

Nếu bạn gặp lỗi Operation not permitted khi tạo các nút OpenThread mới (bằng lệnh mknod), hãy đảm bảo rằng bạn đang chạy Docker với tư cách là người dùng gốc theo các lệnh có trong lớp học lập trình này. Lớp học lập trình này không hỗ trợ chạy Docker ở chế độ gốc.

7. Xin chúc mừng!

Bạn đã mô phỏng thành công mạng Thread đầu tiên của bạn bằng OpenThread. Tuyệt vời!

Trong Lớp học lập trình này, bạn đã tìm hiểu cách:

  • Bắt đầu và quản lý vùng chứa OpenThread Simulation Docker
  • Mô phỏng một mạng Thread
  • Xác thực các nút Luồng
  • Quản lý mạng Thread bằng OpenThread Daemon

Để tìm hiểu thêm về Thread và OpenThread, hãy khám phá các tài liệu tham khảo sau:

Hoặc hãy thử dùng OpenThread Border Router trong vùng chứa Docker!