bbfdm/utilities
Xiaofeng Meng bb85e2c7df
adaption for dm-framework
(cherry picked from commit a9074206cb)

Co-authored-by: Xiaofeng Meng <x.meng@genexis.eu>
2025-10-24 11:09:02 +00:00
..
files/usr Multiple updates 2025-09-14 12:10:52 +05:30
src/ubus adaption for dm-framework 2025-10-24 11:09:02 +00:00
Makefile Register external handler 2025-09-11 13:29:19 +05:30
README.md Multiple updates 2025-09-14 12:10:52 +05:30

Utilities to support datamodel deployments

This directory has sets of small utilities to ease the datamodel deployments and integration with higher layer application protocols.

List of utilities:

  1. bbf.diag
  2. bbf.secure
  3. bbf_configd

bbf.diag utility

bbf.diag is a rpcd libexec based utility, which is responsible for providing ubus backend to few datamodel diagnostics commands like:

  • ipping
  • nslookup
  • serverselection
  • udpecho

How to add a new bbf.diag utility

To add a new bbf.diag utility user has to add an script with below syntax

#!/bin/sh

. /usr/share/libubox/jshn.sh
. /usr/share/bbfdm/scripts/bbf_api

__list() {
        json_add_object "test"
        json_add_string "host" "str"
        json_close_object
}

__error() {
        json_init
        json_add_string "Status" "$1"
}

__launch() {
        json_init
        json_add_string "Status" "$1"
}

if [ "$1" = "list" ]; then
        __list
elif [ -n "$1" ]; then
        __launch "$1"
else
        __error "Error_Internal" "1"
fi

User needs to install this newly added script using BBFDM_INSTALL_SCRIPT bbfdm.mk API.

References:

bbf.secure utility

USPTR369 Specifications requirement of 'SecuredRole' requires bbfdm layer to retrieve and provide the Secured(Password/key) parameter values in plain text.

But storing Secured parameters in plain text in uci/filesystem is bit of a security risk, to mitigate the risk bbf.secure provides way to store the Secured parameter values in hashed format.

A how to use guide for bbf.secure available here

bbf_configd

The BBF Config daemon (bbf_configd) is a configuration management system designed for device management. It provides UBUS-based configuration commit and revert operations with service monitoring and reload capabilities.

OpenWRT way of reloading services with ubus call uci commit '{"name":"<uci_name>"} does not perfectly fits with datamodel requirements. It send a trigger via rpcd to procd by using this ubus call which returns instantly, internally procd reloads for all the services which has a reload dependency configured on that specific uci. Sometimes, there is a good amount of delay in trigger and actual service reload. bbf_configd solves that by adding an in-build reload monitoring functionality which get the list of impacted services and then monitor for PID change for the services with a timeout of 10 sec, with this we make sure Higher layer application(icwmp/obsupa) waits for the application before processing more command.

Architecture

  • Purpose: Daemon implementing UBUS interface for configuration management
  • Key Features:
    • Configuration commit/revert operations
    • Service monitoring and validation
    • Event handling for configuration changes
    • Critical service detection and monitoring

UBUS Interface

Methods

commit

"commit":{"services":"Array","proto":"String","reload":"Boolean"}

  • Description: Commits configuration changes and reloads affected services
  • Parameters:
    • services (array): List of specific services to commit
    • proto (string): Protocol type ("both", "cwmp", "usp")
    • reload (bool): Whether to reload services after commit
  • Response: Status message indicating success or failure
revert

"revert":{"services":"Array","proto":"String","reload":"Boolean"}

  • Description: Reverts uncommitted configuration changes
  • Parameters:
    • services (array): List of specific services to revert
    • proto (string): Protocol type
  • Response: Status message indicating success or failure

Events

bbfdm.apply
  • Trigger: Sent after configuration commits
  • Payload: Protocol information and list of changed UCI files
  • Purpose: Notifies components about applied configuration changes to perform synchronization if required

Configuration Management

Protocol Support

The daemon supports multiple protocols with dedicated configuration directories:

Protocol Config Directory DMMAP Directory Index
both /tmp/bbfdm/.bbfdm/config/ /tmp/bbfdm/.bbfdm/dmmap/ 0
cwmp /tmp/bbfdm/.cwmp/config/ /tmp/bbfdm/.cwmp/dmmap/ 1
usp /tmp/bbfdm/.usp/config/ /tmp/bbfdm/.usp/dmmap/ 2

UCI Integration

  • Standard UCI Path: /etc/config/
  • DMMAP UCI Path: /etc/bbfdm/dmmap/
  • Operations: Commit, revert, and list configurations
  • Persistence: Changes saved to protocol-specific directories

Service Management

Service Monitoring

  • Validation: Checks if services are properly reloaded after configuration changes
  • Criteria:
    • Instance status changes (running/stopped)
    • Process ID (PID) changes
    • Instance count changes
  • Timeout: Configurable wait time for service reload validation

Critical Services

  • Definition File: /etc/bbfdm/critical_services.json
  • Purpose: Identifies services that require monitoring during configuration changes
  • Effect: Enables asynchronous request handling with validation

External Handlers

  • Configuration Path: /etc/bbfdm/services/
  • Types: DMMAP and UCI handlers
  • Default Handler: /etc/bbfdm/bbf_default_reload.sh
  • Purpose: Custom scripts for service-specific configuration handling
Example and Explanation
{
  "daemon": {
    "enable": "1",
    "service_name": "xxxx",
    "unified_daemon": true,
    "services": [
      {
        "parent_dm": "Device.",
        "object": "XXXX"
      }
    ],
    "config": {
      "loglevel": "3"
    },
    "apply_handler": {
      "uci": [
        {
          "file": [
            "xxxx",
            "yyyy"
          ],
          "external_handler": "/etc/xxxx/bbf_config_reload.sh"
        }
      ],
      "dmmap": [
        {
          "file": [
            "zzzz"
          ],
          "external_handler": "/etc/xxxx/bbf_dmmap_handler.sh"
        }
      ]
    }
  }
}
  • If external component asks to commit UCI file 'xxxx' or 'yyyy' or both then this module will commit the changes from protocol based UCI save directory to the standard UCI and will execute the script provided in the external_handler field. If no external handler is provided then it will execute default reload handler script /etc/bbfdm/bbf_default_reload.sh.
  • If there is any change in the dmmap file zzzz and a commit request has been received then this module will commit the changes in standard dmmap file from the protocol based dmmap directory and if any external handler is given like in above example then it will execute that script. Services may use this handler script to perform any desired tasks on changes in the dmmap file.

Operational Flow

Commit Operation

  1. Parse incoming UBUS request parameters
  2. Load critical services configuration
  3. If monitoring required, capture current service states
  4. Process configuration changes:
    • Specific services: Process only requested services
    • All services: Process all modified configurations
  5. Execute external handlers for affected services if any otherwise default handler
  6. If monitoring enabled:
    • Defer response and validate service reloads
    • Send response after validation or timeout
  7. Send bbfdm.apply event with list of all modified UCI names
  8. Clean up allocated resources

Revert Operation

  1. Parse incoming UBUS request parameters
  2. Revert UCI changes for specified or all services
  3. Return immediate success response

Flow Diagram

flowchart TD
    A[Start bbf_configd] --> B[Parse command line args]
    B --> C[Initialize ulog & uloop]
    C --> D[Connect to UBUS]
    D --> E[Load critical services JSON]
    E --> F[Load apply handlers from /etc/bbfdm/services]
    F --> G[Register UBUS object 'bbf.config']
    G --> H[Register event handler 'bbf.config.notify']
    H --> I[Enter uloop_run - Wait for requests]

    I --> J{Incoming Request}

    J -->|commit| K[bbf_config_commit_handler]
    J -->|revert| L[bbf_config_revert_handler]
    J -->|notify event| M[receive_notify_event]

    %% Commit Handler Flow
    K --> K1[Parse blob message]
    K1 --> K2[Allocate async request structure]
    K2 --> K3[Determine protocol index from 'proto']
    K3 --> K4[Get monitor status for critical services]
    K4 --> K5{Monitor required?}

    K5 -->|Yes| K6[Get current service info via UBUS]
    K5 -->|No| K7[Process services]

    K6 --> K7
    K7 --> K8{Services specified?}

    K8 -->|Yes| K9[commit specified configs]
    K8 -->|No| K10[commit all config changes]

    K9 --> K11[Execute action scripts]
    K10 --> K11
    K11 --> K12{Monitor enabled?}

    K12 -->|Yes| K13[Defer request & set timeout]
    K12 -->|No| K14[Send immediate reply]

    K13 --> K15[validate_required_services]
    K15 --> K16{Services reloaded?}
    K16 -->|Yes| K17[Complete request, send reply]
    K16 -->|No| K18[Wait & retry]
    K18 --> K17

    K14 --> K20[Send bbfdm.apply event]
    K17 --> K19[Sync with externally changed uci list]
    K19 --> K20
    K20 --> I

    %% Revert Handler Flow
    L --> L1[Parse blob message]
    L1 --> L2[Determine protocol index]
    L2 --> L3{Services specified?}
    L3 -->|Yes| L4[revert specified_services]
    L3 -->|No| L5[revert all services]
    L4 --> L6[Send reply]
    L5 --> L6
    L6 --> I

    %% Notify Event Flow
    M --> M1[Parse config name from event]
    M1 --> M2{Internal commit in progress?}
    M2 -->|Yes| M3[Add to global external changed list]
    M2 -->|No| M4[Send immediate bbfdm.apply event]
    M3 --> I
    M4 --> I

    %% Service Processing Detail
    subgraph SP[Service Processing Details]
        SP1[commit services] --> SP2[For each service]
        SP2 --> SP3[Determine UCI path & save directory]
        SP3 --> SP4[Set UCI context directories]
        SP4 --> SP5[Lookup UCI pointer]
        SP5 -->|Commit| SP6[uci_commit]
        SP6 --> SP7[Add specific action to action list]
        SP7 --> SP8[Add to changed UCI list if not DMMAP]
    end

    %% UBUS Service Info Gathering
    subgraph SG[Service Info Gathering]
        SG1[fill_service_info] --> SG2[Call service.list via UBUS]
        SG2 --> SG3[_get_service_list_cb or _get_specific_service_cb]
        SG3 --> SG4[Parse service instances & triggers]
        SG4 --> SG5[Build config package structure]
    end

    K6 -.-> SG1
    K15 -.-> SG1
    K9 -.-> SP1
    K10 -.-> SP1

    style K fill:#e1f5fe
    style L fill:#f3e5f5
    style M fill:#e8f5e8
    style K13 fill:#fff3e0
    style K20 fill:#fce4ec

Error Handling

Common Error Scenarios

  • Memory Allocation Failures: Proper cleanup and error responses
  • UCI Operation Failures: Logging and graceful continuation
  • UBUS Communication Errors: Timeout handling and retries
  • Service Validation Timeouts: Fallback completion after maximum wait time

Configuration Files

Critical Services Definition (critical_services.json)

{
  "cwmp": ["cwmp_service"],
  "usp": ["usp_service"]
}
Example
cat /etc/bbfdm/critical_services.json 
{
        "usp": [
                        "/etc/config/mapcontroller",
                        "/etc/config/wireless",
                        "/etc/bbfdm/dmmap/WiFi",
			...
			...
        ],
        "cwmp": [
                        "/etc/config/mapcontroller",
                        "/etc/config/wireless",
                        "/etc/bbfdm/dmmap/WiFi",
                        ...
                        ...
        ]
}

Service Configuration for apply handler (services/*.json)

{
  "daemon": {
    "enable": "0 or 1",
    "service_name": "name_of_the_service",
    ...
    ...
    "apply_handler": {
      "uci": [
        {
          "file": ["config_name"],
          "external_handler": "/path/to/handler.sh"
        }
      ],
      "dmmap": [
        {
          "file": ["dmmap_file"],
          "external_handler": "/path/to/dmmap_handler.sh"
        }
      ]
    }
  }
}

Usage

UBUS Commands

# Commit all changes
ubus call bbf.config commit '{"proto":"both", "reload":true}'

# Commit specific services
ubus call bbf.config commit '{"services":["network","wireless"], "proto":"both"}'

# Revert all changes
ubus call bbf.config revert '{"proto":"both"}'

# Revert specific services
ubus call bbf.config revert '{"services":["network"], "proto":"both"}'

Dependencies

Libraries

  • libubox: UBUS and event loop functionality
  • libuci: UCI configuration management
  • json-c: JSON parsing for configuration files
  • libc: Standard C library functions

System Requirements

  • OpenWrt-based system
  • UBUS daemon running
  • UCI system configured
  • Proper directory structure for configuration files