うたカモ技術ブログ

Linux OpenWrt

OpenWrtアプリケーション開発   自作スクリプトと連携するLuCI用プラグインの作成と追加(難易度Lv2)

post:     update: 

この記事で紹介するソースコードは自由に使っていただいて構いません。 アプリケーションの開発や自己学習にお役立て下さい。ただし、当ブログでは掲載するソースコードを流用・利用したことによる損害等につきましては 一切の責任を負いません。自己責任で利用お願いします。

この記事では、OpenWrtアプリケーション開発として、LuCI用プラグインの作成方法を紹介します。

今回はプラグイン作成方法の1つ(※)として、UBUSシステムのサーバーアプリケーションである/usr/bin/rpcd(以下、rpcd)に自作したスクリプトを 読み込ませる方法について紹介します。

これによって、プロセス間通信機能(IPC)を持たない通常のスクリプトをUBUSシステム経由で実行でき、UBUSシステム のインタフェースをWebUIのLuCIに提供することでプラグインを実現することができます。

rpcdが読み込めるスクリプトは主にAshスクリプトとLuaスクリプトの2種類です。 これらのスクリプトはrpcdに正しく読み込んでもらうための記述ルールに則っている必要があります。

この記事では、これらスクリプトのrpcd読み込み用テンプレートを紹介し、実際に動作するサンプルコードを作成していきます。

この記事を読んで恩恵がある読者は以下のような方です。

  1. カスタマイズ機能をLuCIに追加したい人、それを配布したい人

興味があれば、ご参考ください。

※もう1つのプラグイン作成方法

もう1つのプラグイン作成方法はC言語によるUBUSサーバーアプリケーション開発です。デバイスドライバーと 直接的な情報のやり取りをしたいときなどはこちらの方法でプラグインを作る必要があります。しかし、実装難易度は高めです。

関連記事について

本記事の関連として以下の記事があります。実装難易度を3段階に分けて紹介しています。

目次

  1. 実施環境と補足事項について
  2. サンプルのダウンロード
  3. 今回作成するLuCIプラグインの概要
  4. 本題:開発準備
  5. 本題:LuCI用プラグインの作成
  6. 本題:プラグイン用WebUIページの作成
  7. 配布方法①:セットアップスクリプトの作成と実行
  8. 配布方法②:プラグインのパッケージ作成とインストール
  9. おわりに
  10. 参考文献

実施環境と補足事項について

この記事は以下の環境で実施した結果を元に作成しています。

#開発用PCの実行環境OS
ubuntu 22.04 LTS 64bit ※今回はUbuntuで開発しましたが、Windows(WSL2)上でもOKです。
#OpenWrtデバイス
  名称:Raspberry Pi3 Model B
  CPU:ARM Cortex-A53 (1.2GHz)
  SOC:Broadcom BCM2837
  RAM:1GB
  ストレージ(media):MicroSDカード
#OpenWrt
  OpenWrt 22.03
パッケージ
  luci - git-24.078.77116-81496b8
  luci-compat - git-24.078.77116-81496b8
  luci-lua-runtime - git-24.078.77116-81496b8

この記事で紹介するプラグインの作成と追加方法に関して、機種依存はありません。 この記事ではRaspberry Pi3BをOpenWrtデバイスとしていますが、どんなデバイスでもOKです。(※ただし、LuCIと上記の依存パッケージがインストールされていることが前提です。)

その他補足次項

記事内で紹介するコンソール表記は次の通りです。

開発用PCのコンソール表記
ユーザー名とカレントディレクトリを色付けして表記します。
user:~/openwrt$ command
OpenWrtのコンソール表記
ユーザー名はroot@OpenWrtです。
root@OpenWrt:~# command

この記事が想定するOpenWrtデバイスの初期状態

この記事で想定するOpenWrtデバイスの状態は、OpenWrtがインストールされた直後の初期状態です。
IPアドレスに着目すれば、OpenWrtデバイス自身は192.168.1.1/24のIPアドレスを持ち、192.168.1.0/24のネットワークを管理しています。

サンプルのダウンロード

この記事で作成するLuCI用プラグインのデータは以下の2通りの方法で入手できます。

セットアップスクリプトで自分のOpenWrtデバイスにインストールする

本記事のこちらを参照してください。

Gitリポジトリをクローンする

今回取り上げるLuCI用プラグイン一式は私のGitHubリポジトリにあります。 パッケージ作成に必要なMakefileも揃った完全版です。

一応、この記事の内容に沿って1つずつ作業をしていけば、このリポジトリにあるものと同じものが作成できます。

記述ミスなどで躓いたら、以下のリポジトリをダウンロードして手っ取り早く動作を確認してみてください。

今回作成するLuCIプラグインの概要

この記事で作成するプラグインは以下の動画で紹介するものです。 インタフェース情報やUCI情報を取得するAshスクリプト(ash-sample)とLuaスクリプト(lua-sample)を作成し、それらのスクリプトの処理結果をJSONとしてブラウザー 側に提供して表示します。

内部構成としては、既存のOpenWrt環境の/usr/libexec/rpcdにプラグインとなるAshスクリプトとLuaスクリプトを追加しているだけです。 rpcdによって、これらのスクリプトをUBUS経由で実行することでインタフェース情報とUCI情報を取得します。

それでは、次節からプラグイン作成を始めましょう。

本題:開発準備

LuCI用プラグインを作成する前に開発準備をしましょう。

準備1:OpenWrtのインストール 

当然ですが、まずはOpenWrtデバイスを用意しましょう。一応、私の記事にはRaspberry Pi1~4を対象にOpenWrtファームウェア をインストールする方法を紹介していますので、良ければ参考にしてみてください。

なお今回、LuCI(WebUI)は依存モジュールとしてluci-compatluci-lua-runtimeを利用しますので、メニューコンフィグの該当項目に チェックを入れてください。

既にOpenWrtをデバイスにインストールした方

LuCIまたはコンソール上でluci-compatとluci-lua-runtimeを追加インストールしてください。 次はコンソール上でのインストール例です。

root@OpenWrt:~# opkg update
root@OpenWrt:~# opkg install luci-compat
root@OpenWrt:~# opkg install luci-lua-runtime <--- 正直、こちらは既にインストールされていると思います。

準備2:OpenWrtデバイスと開発用PCを接続してコンソールにログインする

OpenWrtデバイスを用意できたら、次の図のようにLANケーブルでOpenWrtデバイスと開発用PCを繋いでください。

※図ではRaspberry PiがOpenWrtデバイスです。読者の方は、ご自身が持つOpenWrtデバイスとして置き換えてみてください。

初期状態のOpenWrtデバイス(192.168.1.1/24)が管理するデフォルトのLANネットワークは192.168.1.0/24であり、SSHサーバーのdropbearが起動するようになっています。

また、SSHログインに必要な初期パスワード設定は無しのため、以下のコマンドを実行することでログインが成功します。

kamo@kamo:~$ ssh 192.168.1.1 -l root

準備3:SCPコマンドによるファイル転送方法を理解する

次に、UbuntuやWindowsで利用可能なSCPコマンドについて使い方を押さえましょう。

SCPコマンドは作成したプラグイン(スクリプトやHTMLファイル)をOpenWrtデバイスに転送するために使用します。

OpenWrtをインストールしたばかりのデバイス条件では、次のようにSCPコマンドを実行することでファイル転送が可能です。

kamo@kamo:~$ touch test-file
kamo@kamo:~$ echo helloworld > test-file
kamo@kamo:~$ scp ./test-file root@192.168.1.1:/root

上記は、開発用PCのカレントディレクトリ内のtest-fileをOpenWrtデバイス(192.168.1.1/24)の/rootディレクトリに転送した例です

これだけ押さえておけば、ひとまず大丈夫かと思います。もっと知りたい方はこちらを参照してみてください。

準備4:LuCIの変数キャッシュ機能をOFFにする

プラグインをSCPコマンドで転送して動作確認をする際に、LuCIのキャッシュ機能が最新Webページの表示を邪魔します。

そのため、LuCIのキャッシュ機能を表すUCIオプション(luci.ccache.enable)の値をOFFを示す0に変更してください。

root@OpenWrt:~# uci set luci.ccache.enable=0
root@OpenWrt:~# uci commit luci
root@OpenWrt:~# reboot

本題:LuCI用プラグインの作成

ここでは、プラグインの本体となるAsh(その1)とLuaスクリプト(その2)の2つを作成します。

その1:インタフェース情報を取得するAshスクリプトの作成

今回作成するAshスクリプトは特定インタフェースが持つIPアドレス、MACアドレスなどの情報を出力する簡単なものです。

ただし、実際にはAshスクリプトがプラグインのエントリーポイントとなり、その下で呼び出されるLuaスクリプト側でインタフェース情報の取得とJSON出力を実現します。

※実処理をLuaスクリプトに任せる理由は、コマンドライン引数として受け取ったパラメーターを元に外部コマンドを実行し、その結果をJSONに整形するのがLuaスクリプト の方がAshスクリプト単体の構成に比べて楽だからです。テーブルデータをJSONに整形してくれるLuCI用Luaモジュールを利用しています。

このAshスクリプトはUBUSサーバーアプリケーションのrpcdに読み込んでもらい、UBUSシステムに機能の公開を委譲します。 そのため、rpcdが読むことができる記述ルールでスクリプトを作成する必要があります。

rpcdが読み込めるAshスクリプトのテンプレートは次のOpenWrt公式ドキュメントに記載されています。

そのため、上記ページに記載された形式を守ったAshスクリプト(ash-sample)を書き、実処理のインタフェース情報の取得とJSON出力を実行するLuaスクリプト(for-ash-sample)を作成すると以下になります。

ash-sample

#!/bin/sh
# ash-sample
 
case "$1" in
	list)
		echo '{ "hello": {}, "active_if_list": {}, "interface_ip4": {"ifname": "str" }, "interface_mac": {"ifname":"str"}}'
	;;
	call)
		case "$2" in
			hello)
				echo '{"reply":"Hello User!!!"}'
			;;
			active_if_list)
				interfaces=$(ip link show up | awk '/^[0-9]+: / {print substr($2, 1, length($2)-1)}')
                    
				json="{\"interfaces\":["
                
				for iface in $interfaces; do
					json="${json}\"${iface}\","
				done
                
				json="${json%,}]}"
            
				echo $json
			;;
			interface_ip4)
				read input
				lua /usr/bin/for-ash-sample.lua "$2" "$input" 
			;;
			interface_mac)
				read input
				lua /usr/bin/for-ash-sample.lua "$2" "$input"
			;;
		esac
	;;
	*)
		echo '{ "error": "Invalid Argument"}'
	;;
esac

for-ash-sample.lua

#!/usr/bin/env lua
# for-ash-sample.lua

local uci = require("luci.model.uci").cursor()
local json = require "luci.jsonc"

local execute = function(cmd)
    local handle = io.popen(cmd) --環境によっては実行できないこともあるそうです
    local result = handle:read('*a')
    handle:close()
    
    return result
end

if #arg ~= 2 then
    print('{ "error": { "reason": "No argument" }}')
    return
end

local method_param = json.parse(arg[2])

local data = {}

if arg[1] == 'interface_ip4' then
    cmd = "ifconfig " .. method_param.ifname .. " | sed -n 's/.*inet addr:\\([0-9\\.]*\\).*/\\1/p' | tr -d '\n'"
    data.addr = execute(cmd)
elseif arg[1] == 'interface_mac' then
    cmd =  "ifconfig " .. method_param.ifname .. " | awk '/HWaddr/ {print $5}' | tr -d '\n'"
    data.mac = execute(cmd)
end

if not next(data) then
    print('{ "error": { "reason": "no result" }}')
    return
end

local json_output = json.stringify(data, false)
print(json_output)

以上により、プラグイン本体(AshスクリプトとLuaスクリプト)の作成は完了です。

最後に、AshスクリプトをOpenWrt環境の/usr/libexec/rpcdに、Luaスクリプトを/usr/binにアップロードします。

kamo@kamo:~$ scp ./ash-sample root@192.168.1.1:/usr/libexec/rpcd
kamo@kamo:~$ scp ./for-ash-sample.lua root@192.168.1.1:/usr/bin

何故、Luaスクリプトを/usr/binに配置したか?

/usr/binに存在する実行ファイルは、カレントディレクトリがどこであろうともファイル名だけで実行できるためです。 通常、インストールされた大半のアプリケーションは/usr/sbinまたは/usr/binに配置されるため、ユーザーがどのディレクトリに いようともファイル名だけ打ち込んでエンターを押せば起動できます。今回も実行ファイルはそれに倣っています。

ディレクトリ:/usr/libexec/rpcdがなければ作成する

/usr/libexec/rpcdがOpenWrt環境になければ、root@OpenWrt:~# mkdir -p /usr/libexec/rpcd で作成してからアップロードしてください。ここに置いたAshスクリプトはUBUSのサーバーアプリケーションとして働く/usr/bin/rpcdに よって認識されます。rpcdがUBUSシステムにAshスクリプトのオブジェクト登録と外部要求に応じた公開機能実行などを代行してくれます。

無事にアップロードできたらスクリプトに実行権限を与え、バックエンドサーバーとなるrpcdを再起動します。

root@OpenWrt:/usr/libexec/rpcd# chmod +x ash-sample
root@OpenWrt:~# service rpcd restart

これでプラグインを実行する準備が整いました。

上記コマンドの実行でプラグインがUBUSシステムに認識されると、次のlistコマンドでUBUSオブジェクトのash-sampleが 表示されるはずです。

それでは、動作確認をしてみます。

プラグインのスクリプト(/usr/libexec/rpcd/ash-sample)の名前がUBUSオブジェクトの名前になります。

root@OpenWrt:~# ubus list
ash-sample   <---私の環境では、ここに表示されました
container
dhcp
dnsmasq
dnsmasq.dns
file
hostapd
hotplug.dhcp
hotplug.ieee80211
hotplug.iface
hotplug.neigh
hotplug.net
hotplug.ntp
hotplug.tftp
iwinfo
log
luci
luci-rpc
network
network.device
network.interface
network.interface.lan
network.interface.loopback
network.rrdns
network.wireless
rc
service
session
system
uci
wpa_supplicant

次のコマンドを実行すると、呼び出し時の引数指定の方法が分かります。

root@OpenWrt:~# ubus -v list ash-sample
'ash-sample' @3e30ae5e
        "hello":{}
        "active_if_list":{}
        "interface_ip4":{"ifname":"String"}
        "interface_mac":{"ifname":"String"}

ここまでで特に問題がなければ、プラグインは正常にUBUSシステムに認識されていると言えます。 ということで、さっそくUBUSのcallコマンドを使用してプラグインを実行してみましょう。

以下はプラグインの実行結果です。

root@OpenWrt:~# ubus call ash-sample hello
{
        "reply": "Hello User!!!"
}
root@OpenWrt:~# ubus call ash-sample active_if_list
{
        "interfaces": [
                "lo",
                "eth0",
                "phy0-ap0",
                "br-lan"
        ]
}
root@OpenWrt:~# ubus call ash-sample interface_ip4 '{"ifname":"br-lan"}'
{
        "addr": "192.168.3.7"
}
root@OpenWrt:~# ubus call ash-sample interface_mac '{"ifname":"br-lan"}'
{
        "mac": "B8:27:EB:B8:6B:03"
}

これで動作確認ができました。

その2:UCI情報を取得するLuaスクリプト

次はLuaスクリプトをプラグインのエントリーポイントとして、任意のUCIコンフィグの オプション・リスト情報を取得する処理を作成します。

このLUaスクリプトはUBUSサーバーアプリケーションのrpcdに読み込んでもらい、UBUSシステムに機能の公開を委譲します。 そのため、rpcdが読むことができる記述ルールでスクリプトを作成する必要があります。

rpcdが読み込めるLuaスクリプトのテンプレートは次のOpenWrtのGitHubページに記載されています。

上記URLのテンプレートをそのまま使っても良いのですが、今回はUCIコンフィグ(例:wireless)を指定すると その中のオプションとリストを表示するスクリプト(lua-sample)を書いてみました。

lua-sample

#!/usr/bin/env lua

local jsonc = require("luci.jsonc")
local uci = require("luci.model.uci").cursor()

local methods = {

    greeting = {
        call = function()
            local r = {}

            -- Write some process

            local data = {}
            data.reply1 = "good morning!!"
            data.reply2 = "Hello!!"
            data.reply3 = "good bye!!"

            r.result = jsonc.stringify(data)

            return r
        end
    },

    echo = {
        args = { arg = "a_string" },

        call = function(args)
            local r = {}

            -- Write some process

            r.result = jsonc.stringify({
                user_input = args.arg
            })

            return r
        end
    },

    config_detail = {
        args = { config = "a_string" },
        
        call = function(args)
            local r = {}
            local data = uci:get_all(args["config"])
            r.result = jsonc.stringify(data, false)
            return r
        end
    },
}

local function parseInput()

    local parse = jsonc.new()
    local done, err

    while true do
        local chunk = io.read(4096)
        if not chunk then
            break
        elseif not done and not err then
            done, err = parse:parse(chunk)
        end
    end

    if not done then
        print(jsonc.stringify({
            error = err or "Incomplete input for argument parsing"
        }))
        os.exit(1)
    end

    return parse:get()
end

-- validation
local function validateArgs(func, uargs)

    local method = methods[func]
    if not method then
        print(jsonc.stringify({error = "Method not found in methods table"}))
        os.exit(1)
    end

    local n = 0
    for _, _ in pairs(uargs) do n = n + 1 end

    if method.args and n == 0 then
        print(jsonc.stringify({
            error = "Received empty arguments for " .. func ..
                " but it requires " .. jsonc.stringify(method.args)
        }))
        os.exit(1)
    end

    uargs.ubus_rpc_session = nil

    local margs = method.args or {}
    for k, v in pairs(uargs) do
        if margs[k] == nil or (v ~= nil and type(v) ~= type(margs[k])) then
            print(jsonc.stringify({
                error = "Invalid argument '" .. k .. "' for " .. func ..
                    " it requires " .. jsonc.stringify(method.args)
            }))
            os.exit(1)
        end
    end

    return method
end

-- ubus list & call
if arg[1] == "list" then
    local _, rv = nil, {}
    for _, method in pairs(methods) do rv[_] = method.args or {} end
    print((jsonc.stringify(rv):gsub(":%[%]", ":{}")))
elseif arg[1] == "call" then
    local args = parseInput()
    local method = validateArgs(arg[2], args)
    local run = method.call(args)
    print(run.result)
    os.exit(run.code or 0)
end

以上により、プラグイン本体の作成は完了です。

最後に、Luaスクリプト(lua-sample)をOpenWrt環境の/usr/libexec/rpcdにアップロードします。

kamo@kamo:~$ scp ./lua-sample root@192.168.1.1:/usr/libexec/rpcd

無事にアップロードできたらスクリプトに実行権限を与え、バックエンドサーバーとなるrpcdを再起動します。

root@OpenWrt:/usr/libexec/rpcd# chmod +x lua-sample
root@OpenWrt:~# service rpcd restart

上記コマンドの実行でプラグインがUBUSシステムに認識されると、次のlistコマンドでUBUSオブジェクトのlua-sampleが 表示されるはずです。

root@OpenWrt:~# ubus list
ash-sample
container
dhcp
file
hostapd
hostapd.phy0-ap0
hotplug.dhcp
hotplug.ieee80211
hotplug.iface
hotplug.neigh
hotplug.net
hotplug.ntp
hotplug.tftp
iwinfo
log
lua-sample  <--- 私の環境では、ここに表示されました
luci
luci-rpc
network
network.device
network.interface
network.interface.lan
network.interface.loopback
network.rrdns
network.wireless
rc
service
session
system
test
uci
wpa_supplicant

そして、次のコマンドを実行すると、呼び出し時の引数指定の方法が分かります。

root@OpenWrt:~# ubus -v list lua-sample
'lua-sample' @45cb3627
        "echo":{"arg":"String"}
        "greeting":{}
        "config_detail":{"config":"String"}

ここまでで特に問題がなければ、プラグインは正常にUBUSシステムに認識されていると言えます。 ということで、さっそくUBUSのcallコマンドを使用してプラグインを実行してみましょう。

以下はプラグインの実行結果です。

root@OpenWrt:/usr/libexec/rpcd# ubus call lua-sample greeting
{
        "reply2": "Hello!!",
        "reply3": "good bye!!",
        "reply1": "good morning!!"
}
root@OpenWrt:/usr/libexec/rpcd# ubus call lua-sample echo '{"arg":"utakamo"}'
{
        "user_input": "utakamo"
}
root@OpenWrt:/usr/libexec/rpcd# ubus call lua-sample config_detail '{"config":"wireless"}'
{
        "radio0": {
                "band": "2g",
                ".anonymous": false,
                "disabled": "0",
                "country": "JP",
                "txpower": "10",
                ".index": 0,
                "path": "platform/soc/3f300000.mmcnr/mmc_host/mmc1/mmc1:0001/mmc1:0001:1",
                "channel": "1",
                ".name": "radio0",
                ".type": "wifi-device",
                "htmode": "HT20",
                "type": "mac80211"
        },
        "default_radio0": {
                ".name": "default_radio0",
                ".anonymous": false,
                "ssid": "OpenWrt",
                "encryption": "psk2",
                "device": "radio0",
                ".index": 1,
                "key": "utakamo1234",
                "mode": "ap",
                ".type": "wifi-iface",
                "network": "lan"
        }
}

これで動作確認ができました。

本題:プラグイン用WebUIページの作成

LuCIのページの作成方法は前回記事の「OpenWrtアプリケーション開発 LuCI用カスタムページの作成と追加(難易度Lv1)」で紹介しましたので、 サクッと作っていきます。

作成するスクリプトは上記のプラグインをWeb経由からの要求に従って実行するmodule.lua、 そして、ユーザーが実際に操作するWebページ(sample_ash_plugin.htmとsample_lua_plugin.htm)の3つです。

module.lua

ここではまず、index関数内で各プラグインのWebUI(Webページ)へのアクセスと、各プラグイン機能を実行するエントリポイントのルーティング設定が記述されています。

index関数内の指定によって、WebUIのアクセスとプラグイン機能実行がすべてWebブラウザー上のURL指定によって実現できます。 例えば、OpenWrtデバイスのIPアドレスが192.168.1.1の場合、以下のURLが利用できます。

アクセス・実行対象 対応URL
Ashスクリプトのプラグインページ https://192.168.1.1/cgi-bin/luci/admin/status/sample_ash_plugin
Luaスクリプトのプラグインページ https://192.168.1.1/cgi-bin/luci/admin/status/sample_lua_plugin
Ashスクリプトのactive_if_list関数 https://192.168.1.1/cgi-bin/luci/admin/status/sample_ash_plugin/active_if_list
Ashスクリプトのinterface_ip4関数 https://192.168.1.1/cgi-bin/luci/admin/status/sample_ash_plugin/interface_ip4
Ashスクリプトのinterface_mac関数 https://192.168.1.1/cgi-bin/luci/admin/status/sample_ash_plugin/interface_mac
Luaスクリプトのconfig_detail関数 https://192.168.1.1/cgi-bin/luci/admin/status/sample_lua_plugin/config_detail

index関数以降はLuaスクリプトのUBUS CALLコマンド実装とそれを利用して呼び出されるプラグイン機能の実行関数が記述されています。

-- module.lua
module("luci.controller.luci-sample-app02.module", package.seeall)

function index()
    entry({"admin", "status", "sample_ash_plugin"}, template("luci-sample-app02/sample_ash_plugin"), _("Sample Ash Plugin"), 90)
    entry({"admin", "status", "sample_lua_plugin"}, template("luci-sample-app02/sample_lua_plugin"), _("Sample Lua Plugin"), 90)
    entry({"admin", "status", "sample_ash_plugin", "active_if_list"}, call("action_sample_ash_plugin_active_if_list"), nil).leaf = true
    entry({"admin", "status", "sample_ash_plugin", "interface_ip4"}, call("action_sample_ash_plugin_interface_ip4"), nil).leaf = true
    entry({"admin", "status", "sample_ash_plugin", "interface_mac"}, call("action_sample_ash_plugin_interface_mac"), nil).leaf = true
    entry({"admin", "status", "sample_lua_plugin", "config_detail"}, call("action_sample_lua_plugin"), nil).leaf = true
end

-- [References] https://openwrt.org/docs/techref/ubus#lua_module_for_ubus
function ubus_call(path, method, param)
    local ubus = require("ubus")
    local conn = ubus.connect()

    if not conn then
        return { error = "Unable to connect to ubus" }
    end

    local result, err = conn:call(path, method, param)
    conn:close()

    if not result then
        return { error = err or "ubus call failed" }
    end

    return result
end

function action_sample_ash_plugin_active_if_list()
    local luci_http = require "luci.http"
    
    -- ubus call
    local result = ubus_call("ash-sample", "active_if_list", {})

    luci_http.prepare_content("application/json")
    luci_http.write_json(result)
end

function action_sample_ash_plugin_interface_ip4()
    local luci_http = require "luci.http"

    -- Get parameters from the HTTP request
    local params = luci_http.formvalue("params")

    if not params then
        luci_http.prepare_content("application/json")
        luci_http.write_json({ error = "Missing params" })
        return
    end

    -- Create a parameter table for ubus call
    local json_param = { ifname = params }
    
    -- ubus call
    local result = ubus_call("ash-sample", "interface_ip4", json_param)

    luci_http.prepare_content("application/json")
    luci_http.write_json(result)
end

function action_sample_ash_plugin_interface_mac()
    local luci_http = require "luci.http"

    -- Get parameters from the HTTP request
    local params = luci_http.formvalue("params")

    if not params then
        luci_http.prepare_content("application/json")
        luci_http.write_json({ error = "Missing params" })
        return
    end

    -- Create a parameter table for ubus call
    local json_param = { ifname = params }
    
    -- ubus call
    local result = ubus_call("ash-sample", "interface_mac", json_param)

    luci_http.prepare_content("application/json")
    luci_http.write_json(result)
end

function action_sample_lua_plugin()
    local luci_http = require "luci.http"
    
    -- Get parameters from the HTTP request
    local params = luci_http.formvalue("params")

    if not params then
        luci_http.prepare_content("application/json")
        luci_http.write_json({ error = "Missing params" })
        return
    end

    -- Create a parameter table for ubus call
    local json_param = { config = params }
    
    -- ubus call
    local result = ubus_call("lua-sample", "config_detail", json_param)

    luci_http.prepare_content("application/json")
    luci_http.write_json(result)
end

sample_ash_plugin.htm

以下はAshスクリプトで作ったプラグインを呼び出すことができるWebUIです。

Ajaxで以下のURLにアクセスし、結果をJSONとして取得して表示します。
※次のURLはOpenWrtデバイスが192.168.1.1の場合の例です。

アクセス・実行対象 対応URL
Ashスクリプトのactive_if_list関数 https://192.168.1.1/cgi-bin/luci/admin/status/sample_ash_plugin/active_if_list
Ashスクリプトのinterface_ip4関数 https://192.168.1.1/cgi-bin/luci/admin/status/sample_ash_plugin/interface_ip4
Ashスクリプトのinterface_mac関数 https://192.168.1.1/cgi-bin/luci/admin/status/sample_ash_plugin/interface_mac
<!-- sample_ash_plugin.htm -->
<%+header%>
<h1>Sample Ash Plugin</h1>

<!-- Active Interface List -->
<h2>Active Interface List</h2>
<div id="activeIfListResult">Loading...</div>

<!-- Form to fetch IPv4 address of an interface -->
<h2>Interface IPv4 Address</h2>
<form id="interfaceIp4Form">
    <label for="ip4Params">Interface Name:</label>
    <input type="text" id="ip4Params" name="params" required>
    <button type="submit">Show IPv4 Address</button>
</form>
<div id="interfaceIp4Result"></div>

<!-- Form to fetch MAC address of an interface -->
<h2>Interface MAC Address</h2>
<form id="interfaceMacForm">
    <label for="macParams">Interface Name:</label>
    <input type="text" id="macParams" name="params" required>
    <button type="submit">Show MAC Address</button>
</form>
<div id="interfaceMacResult"></div>

<script type="text/javascript">
    document.addEventListener('DOMContentLoaded', function() {
        // Fetch active interface list on page load
        var resultDiv = document.getElementById('activeIfListResult');
        resultDiv.innerHTML = 'Loading...'; // Show loading message

        fetch('<%=build_url("admin", "status", "sample_ash_plugin", "active_if_list")%>', {
            method: 'POST'
        })
        .then(response => response.json())
        .then(data => {
            displayInterfaceList('activeIfListResult', data);
        })
        .catch(error => {
            console.error('Error:', error);
            resultDiv.innerHTML = 'Error: ' + error;
        });
    });

    // Event listener for IPv4 address form
    document.getElementById('interfaceIp4Form').addEventListener('submit', function(event) {
        event.preventDefault();
        var params = document.getElementById('ip4Params').value;
        fetch('<%=build_url("admin", "status", "sample_ash_plugin", "interface_ip4")%>', {
            method: 'POST',
            headers: {
                'Content-Type': 'application/x-www-form-urlencoded'
            },
            body: new URLSearchParams({ params: params })
        })
        .then(response => response.json())
        .then(data => {
            displayResult('interfaceIp4Result', data);
        })
        .catch(error => {
            console.error('Error:', error);
            document.getElementById('interfaceIp4Result').innerHTML = 'Error: ' + error;
        });
    });

    // Event listener for MAC address form
    document.getElementById('interfaceMacForm').addEventListener('submit', function(event) {
        event.preventDefault();
        var params = document.getElementById('macParams').value;
        fetch('<%=build_url("admin", "status", "sample_ash_plugin", "interface_mac")%>', {
            method: 'POST',
            headers: {
                'Content-Type': 'application/x-www-form-urlencoded'
            },
            body: new URLSearchParams({ params: params })
        })
        .then(response => response.json())
        .then(data => {
            displayResult('interfaceMacResult', data);
        })
        .catch(error => {
            console.error('Error:', error);
            document.getElementById('interfaceMacResult').innerHTML = 'Error: ' + error;
        });
    });

    // Function to display active interface list in a table
    function displayInterfaceList(elementId, data) {
        var resultDiv = document.getElementById(elementId);
        resultDiv.innerHTML = ''; // Clear previous results

        var table = document.createElement('table');
        table.className = 'table'; // Use LuCI table CSS class

        var thead = document.createElement('thead');
        var headerRow = document.createElement('tr');
        headerRow.className = 'tr table-titles';

        var headers = ['Interface'];
        headers.forEach(function(text) {
            var th = document.createElement('th');
            th.className = 'th';
            th.appendChild(document.createTextNode(text));
            headerRow.appendChild(th);
        });
        thead.appendChild(headerRow);
        table.appendChild(thead);

        var tbody = document.createElement('tbody');
        data.interfaces.forEach(function(interfaceName) {
            var row = document.createElement('tr');
            row.className = 'tr';

            var cell = document.createElement('td');
            cell.className = 'td';
            cell.appendChild(document.createTextNode(interfaceName));
            row.appendChild(cell);

            tbody.appendChild(row);
        });
        table.appendChild(tbody);
        resultDiv.appendChild(table);
    }

    // Function to display result in a table
    function displayResult(elementId, data) {
        var resultDiv = document.getElementById(elementId);
        resultDiv.innerHTML = ''; // Clear previous results

        var table = document.createElement('table');
        table.className = 'table'; // Use LuCI table CSS class

        var thead = document.createElement('thead');
        var headerRow = document.createElement('tr');
        headerRow.className = 'tr table-titles';

        var headers = ['Key', 'Value'];
        headers.forEach(function(text) {
            var th = document.createElement('th');
            th.className = 'th';
            th.appendChild(document.createTextNode(text));
            headerRow.appendChild(th);
        });
        thead.appendChild(headerRow);
        table.appendChild(thead);

        var tbody = document.createElement('tbody');
        for (var key in data) {
            if (data.hasOwnProperty(key)) {
                var row = document.createElement('tr');
                row.className = 'tr';

                var keyCell = document.createElement('td');
                keyCell.className = 'td';
                keyCell.appendChild(document.createTextNode(key));
                row.appendChild(keyCell);

                var valueCell = document.createElement('td');
                valueCell.className = 'td';
                valueCell.appendChild(document.createTextNode(JSON.stringify(data[key])));
                row.appendChild(valueCell);

                tbody.appendChild(row);
            }
        }
        table.appendChild(tbody);
        resultDiv.appendChild(table);
    }
</script>
<%+footer%>

sample_lua_plugin.htm

以下はLuaスクリプトで作ったプラグインを呼び出すことができるWebUIです。

Ajaxで以下のURLにアクセスし、結果をJSONとして取得してテーブル表示します。
※次のURLはOpenWrtデバイスが192.168.1.1の場合の例です。

アクセス・実行対象 対応URL
Luaスクリプトのconfig_detail関数 https://192.168.1.1/cgi-bin/luci/admin/status/sample_ash_plugin/config_detail
<!-- sample_lua_plugin.htm -->
<%+header%>
<h1>Sample Lua Plugin</h1>
<form id="sampleForm">
    <label for="params">UCI CONFIG NAME:</label>
    <input type="text" id="params" name="params" required>
    <br>
    <button type="submit">Execute</button>
</form>
<div id="result"></div>

<script type="text/javascript">
    document.getElementById('sampleForm').addEventListener('submit', function(event) {
        event.preventDefault();
        var params = document.getElementById('params').value;

        fetch('<%=build_url("admin", "status", "sample_lua_plugin", "config_detail")%>', {
            method: 'POST',
            headers: {
                'Content-Type': 'application/x-www-form-urlencoded'
            },
            body: new URLSearchParams({ params: params })
        })
        .then(response => response.json())
        .then(data => {
            var resultDiv = document.getElementById('result');
            resultDiv.innerHTML = ''; // Clear previous results
            resultDiv.appendChild(generateTable(data));
        })
        .catch(error => {
            console.error('Error:', error);
            document.getElementById('result').innerHTML = 'Error: ' + error;
        });
    });

    function generateTable(data) {
        var table = document.createElement('table');
        table.className = "table"; // Use LuCI table CSS class

        // Create table header
        var thead = document.createElement('thead');
        var headerRow = document.createElement('tr');
        headerRow.className = "tr table-titles";
        ['Name', 'Type', 'Properties'].forEach(function(text) {
            var th = document.createElement('th');
            th.className = "th";
            th.appendChild(document.createTextNode(text));
            headerRow.appendChild(th);
        });
        thead.appendChild(headerRow);
        table.appendChild(thead);

        // Create table body
        var tbody = document.createElement('tbody');
        for (var key in data) {
            if (data.hasOwnProperty(key)) {
                var row = document.createElement('tr');
                row.className = "tr";

                // Name column
                var nameCell = document.createElement('td');
                nameCell.className = "td";
                nameCell.appendChild(document.createTextNode(key));
                row.appendChild(nameCell);

                // Type column
                var typeCell = document.createElement('td');
                typeCell.className = "td";
                typeCell.appendChild(document.createTextNode(data[key][".type"] || ''));
                row.appendChild(typeCell);

                // Properties column
                var propertiesCell = document.createElement('td');
                propertiesCell.className = "td";
                var propertiesTable = document.createElement('table');
                propertiesTable.className = "table"; // Use LuCI table CSS class

                for (var prop in data[key]) {
                    if (data[key].hasOwnProperty(prop) && prop !== '.name' && prop !== '.type') {
                        var propRow = document.createElement('tr');
                        propRow.className = "tr";
                        var propNameCell = document.createElement('td');
                        propNameCell.className = "td";
                        propNameCell.appendChild(document.createTextNode(prop));
                        var propValueCell = document.createElement('td');
                        propValueCell.className = "td";
                        propValueCell.appendChild(document.createTextNode(JSON.stringify(data[key][prop])));
                        propRow.appendChild(propNameCell);
                        propRow.appendChild(propValueCell);
                        propertiesTable.appendChild(propRow);
                    }
                }
                propertiesCell.appendChild(propertiesTable);
                row.appendChild(propertiesCell);

                tbody.appendChild(row);
            }
        }
        table.appendChild(tbody);

        return table;
    }
</script>
<%+footer%>
    

作成したスクリプトとファイルをOpenWrtデバイス環境に配置する

上記のスクリプトとファイルをOpenWrtデバイスに配置します。

最初に次のコマンドを実行してOpenWrtデバイスにディレクトリを作成します。

root@OpenWrt:~# mkdir -p /usr/lib/lua/luci/controller/luci-sample-app02
root@OpenWrt:~# mkdir -p /usr/lib/lua/luci/view/luci-sample-app02

そして、次のSCPコマンドで各スクリプトとファイルをOpenWrtデバイスの対象ディレクトリにアップロードします。

kamo@kamo:~$ scp ./module.lua root@192.168.1.1:/usr/lib/lua/luci/controller/luci-sample-app02
kamo@kamo:~$ scp ./sample_ash_plugin.htm root@192.168.1.1:/usr/lib/lua/luci/view/luci-sample-app02
kamo@kamo:~$ scp ./sample_lua_plugin.htm root@192.168.1.1:/usr/lib/lua/luci/view/luci-sample-app02

これでプラグインを実行するのに必要なものがすべて揃いました。

プラグインの動作確認

それでは動作確認をしてみます。

Ashスクリプトのプラグイン動作確認
まず、LuCIにアクセスにしてWebUIのトップページを開きます。 そして、上部メニューバーの「Status」を開き、「Sample Ash Plugin」をクリックします。

Ashスクリプトのプラグインページが遷移し、OpenWrtデバイスが認識しているインタフェースが テーブル表示されます。

このテーブル表示の内容を参考に、次のIPアドレスとMACアドレスの照会フォームにインタフェースを入力して 「Show Ip Address」、「Show Mac Address」を押します。

すると、次のようにインタフェースが持つIPv4アドレスとMacアドレスが表示されます。 ※この時、存在しない場合はnoneと表示されます。

Luaスクリプトのプラグイン動作確認
LuCIのWebUIトップページを開き、上部メニューバーから「Status」--->「Sample Lua Plugin」をクリックします。

Luaスクリプトのプラグインページに遷移し、UCIコンフィグ名を入力するフォームが表示されます。

この入力フォームにOpenWrtデバイスが管理するUCIコンフィグの名前(例:network)を入力して、Showボタンを押します。

すると、次のようにnetworkコンフィグが持つ設定がテーブル表示されます。

これで、プラグインの動作確認ができました。

配布方法①:セットアップスクリプトの作成と実行

今回作成したプラグインは、特定のハードウェアに依存したものではありません。

そのため、配布方法②で紹介するようなインストールパッケージを用意しなくてもOpenWrtデバイスに適用することができます。

ハードウェアに依存しないモジュールを配布する場合はここで紹介する方法が圧倒的に便利です。

それでは作成していきましょう。

Webサーバーの配布用URL(リポジトリ)を決定する

前節で作成したプラグインを配布するために、まずはサーバーを用意します。

私の場合、GitHubでソース管理をしていますので、GitHubをサーバー代わりにします。

GitHubのリポジトリを作成し、プラグイン一式を収録したアーカイブファイルと それをダウンロードするスクリプト(セットアップスクリプト)をアップロードします。

今回、私は次のURLで示すGitHubリポジトリを用意しました。

https://github.com/utakamo/UtakamoStudyApps/raw/main/webui/luci-sample-app02/setup

このリポジトリに対して、次に説明するアーカイブファイルとセットアップスクリプトをアップロードしていきます。

アーカイブファイルの作成

最初に、各プラグイン用のLuaスクリプトとHTMLファイルを次のディレクトリ・ファイル構成になるように整理します。

上記の構成からアーカイブファイルを作成するには、親のディレクトリを指定して次のコマンドを実行します。

kamo@kamo:~$ tar zcvf luci-sample-app02.tar.gz ./luci-sample-app02
./luci-sample-app02/
./luci-sample-app02/src/
./luci-sample-app02/src/usr/
./luci-sample-app02/src/usr/libexec/
./luci-sample-app02/src/usr/libexec/rpcd/
./luci-sample-app02/src/usr/libexec/rpcd/ash-sample
./luci-sample-app02/src/usr/libexec/rpcd/lua-sample
./luci-sample-app02/src/usr/bin/
./luci-sample-app02/src/usr/bin/for-ash-sample.lua
./luci-sample-app02/webui/
./luci-sample-app02/webui/controller/
./luci-sample-app02/webui/controller/module.lua
./luci-sample-app02/webui/view/
./luci-sample-app02/webui/view/sample_ash_plugin.htm
./luci-sample-app02/webui/view/sample_lua_plugin.htm

作成できたアーカイブファイル(luci-sample02-app.tar.gz)は次で説明するセットアップスクリプトを使って展開し、 その中のファイルとスクリプトをOpenWrtデバイスの環境に配置します。

このアーカイブファイルは上記で説明したGitHubリポジトリにアップロードします。

これによって、このアーカイブは以下のURLでダウンロードが可能です。

https://github.com/utakamo/UtakamoStudyApps/raw/main/webui/luci-sample-app02/setup/luci-sample-app02.tar.gz

セットアップスクリプト(setup-sample02)の作成

最後にセットアップスクリプトを作成します。

上記で作成したアーカイブファイルをOpenWrtデバイスのローカル環境にダウンロードして、 展開をした後に適切なディレクトリに各ファイルをコピーするだけの簡単なスクリプトです。

#/bin/sh
#setup-sample02

DOWNLOAD_URL="https://github.com/utakamo/UtakamoStudyApps/raw/main/webui/luci-sample-app02/setup/luci-sample-app02.tar.gz"
TEMP_ARCHIVE_1="/tmp/luci-sample-app02.tar.gz"
TEMP_ARCHIVE_2="/tmp/luci-sample-app02.tar"
EXTRACT_DIR="/tmp/luci-sample-app02"
WEBUI_SRC_DIR="$EXTRACT_DIR/webui"
WEBUI_DST_DIR="/usr/lib/lua/luci"
PLUGIN_SRC_DIR="$EXTRACT_DIR/src/"
PLUGIN_DST_DIR="/usr/libexec/rpcd"
USR_BIN_DIR="/usr/bin"

help() {
    echo "setup-sample02 install ... install luci-sample-app02"
    echo "setup-sample02 remove  ... remove luci-sample-app02"
}

delete_archive_file() {
    rm -rf "$TEMP_ARCHIVE_1"
    rm -rf "$TEMP_ARCHIVE_2"
    rm -rf "$EXTRACT_DIR"
}

remove_install_file() {
    rm -rf "$WEBUI_DST_DIR/view/luci-sample-app02/"
    rm -rf "$WEBUI_DST_DIR/controller/luci-sample-app02/"
}

install() {

    if ! opkg list-installed luci-compat | grep -q "luci-compat"; then
        echo -n "Do you want to install the luci-compat package? (Y/N) :"
        read reply

        if [ "$reply" = "Y" ]; then
            opkg update
            opkg install luci-compat
        else
            echo "luci-sample-app02 install cancelled."
            exit 1
        fi
    fi

    # download
    wget --no-check-certificate -O "$TEMP_ARCHIVE_1" "$DOWNLOAD_URL"
    if [ "$?" -ne 0 ]; then
        echo "Failed to download $DOWNLOAD_URL" >&2
        exit 1
    fi

    # expands
    mkdir -p "$EXTRACT_DIR"
    gunzip -c "$TEMP_ARCHIVE_1" | tar -x -C /tmp

    if [ "$?" -ne 0 ]; then
    echo "Failed to extract $TEMP_ARCHIVE_1" >&2
    delete_archive_file
        exit 1
    fi

    # remove previous file for reinstall
    remove_install_file

    # create dir and copy
    mkdir -p "$WEBUI_DST_DIR/view/luci-sample-app02"
    mkdir -p "$WEBUI_DST_DIR/controller/luci-sample-app02"

    # plugin script
    mkdir -p "$PLUGIN_DST_DIR/"
    cp -a "$PLUGIN_SRC_DIR/usr/libexec/rpcd/ash-sample" "$PLUGIN_DST_DIR/"
    cp -a "$PLUGIN_SRC_DIR/usr/bin/for-ash-sample.lua" "$USR_BIN_DIR/"
    cp -a "$PLUGIN_SRC_DIR/usr/libexec/rpcd/lua-sample" "$PLUGIN_DST_DIR/"
    chmod +x "$PLUGIN_DST_DIR/ash-sample"
    chmod +x "$USR_BIN_DIR/for-ash-sample.lua"
    chmod +x "$PLUGIN_DST_DIR/lua-sample"

    # webui
    cp -a "$WEBUI_SRC_DIR/controller/module.lua" "$WEBUI_DST_DIR/controller/luci-sample-app02/"
    cp -a "$WEBUI_SRC_DIR/view/sample_ash_plugin.htm" "$WEBUI_DST_DIR/view/luci-sample-app02/"
    cp -a "$WEBUI_SRC_DIR/view/sample_lua_plugin.htm" "$WEBUI_DST_DIR/view/luci-sample-app02/"

    delete_archive_file
    echo "Setup completed successfully"

    echo -n "System Reboot? (Y/N) :"
    read reboot

    if [ "$reboot" = Y ]; then
        reboot
    fi
}

remove() {
    remove_install_file
    echo "Remove completed successfully"
}

if [ "$1" = "install" ]; then
        install
elif [ "$1" = "remove" ]; then
        remove
else
        help
fi    

このセットアップスクリプトも作成後は、リポジトリにアップロードします。

そして、アップロードすると今回は以下のURLでダウンロードできるようになりました。

https://raw.githubusercontent.com/utakamo/UtakamoStudyApps/main/webui/luci-sample-app02/setup/setup-sample02

後はユーザーに、このセットアップスクリプトの使用方法をアナウンスすればいいだけです。

ということで、次はこのスクリプトを使ってOpenWrtデバイスのLuCIにプラグインを適用してみます。

セットアップスクリプトを利用してOpenWrtデバイスにプラグインを適用する

それでは、実際にセットアップスクリプトをリポジトリから入手してプラグインを OpenWrtデバイスのLuCIに適用してみます。

プラグインを適用したいOpenWrtデバイスのコンソール上で次のコマンドを実行して、 セットアップスクリプト(setup-sample02)をダウンロードします。

root@OpenWrt~:# wget https://raw.githubusercontent.com/utakamo/UtakamoStudyApps/main/webui/luci-sample-app02/setup/setup-sample02

時刻情報の不一致で、SSL認証エラーが発生してダウンロードできない場合の対処

OpenWrtデバイス側でwgetコマンドを実行すると、SSL認証に失敗してダウンロードができない場合があります。 大体の原因は時刻情報なので、dateコマンドを実行して現在時刻が正しいか確認してください。

root@OpenWrt:~# date

システムが認識している現在時刻が正しくない場合は、次のコマンドでNTPサーバーから時刻を取得してください。

/etc/init.d/sysntpd restart

これでも、wgetコマンドでダウンロードできない場合は-no-check-certificateオプションを付けて実行してください。

wget --no-check-certificate https://raw.githubusercontent.com/utakamo/UtakamoStudyApps/main/webui/luci-sample-app02/setup/setup-sample02

次に、このセットアップスクリプトを実行します。

root@OpenWrt:~# chmod +x setup-sample02
root@OpenWrt~:# ./setup-sample02 install
Setup completed successfully
System Reboot? (Y/N) :Y

これでプラグインの適用が完了しました。

システムブート後にプラグインの各URLにアクセスしてみると、正常にページが表示されることが確認できます。

配布方法②:プラグインのパッケージ作成とインストール

作成中

おわりに

今回はOpenWrtのLuCI用プラグインの作成方法について紹介しました。

この記事では、UCIコンフィグの情報を取得するだけの簡単なプラグインを作った だけですが、もっと有用な処理をすれば、便利なプラグインとして他のユーザーに 重宝されるかもしれません。

作りたい機能がある方は、実装の際に参考にしてみてください。

参考文献