diff --git a/docs/ucode.css b/docs/ucode.css index a40f4787..966228a6 100644 --- a/docs/ucode.css +++ b/docs/ucode.css @@ -6,6 +6,24 @@ h1 { display: flex; } +h3[id].name:not(h2 + h3) { + margin-top: 2.5rem; + border-top: 1px solid #292929; + padding-top: 1.5rem; +} + +h3[id].name:not(h2 + h3) .link-anchor { + top: 1.5rem; +} + +.light h3[id].name { + border-color: #ccc; +} + +html body { + line-height: 1.5; +} + article .description > ul, article .description > ol { list-style: initial; @@ -18,3 +36,66 @@ article .description > ol { background-position: 1rem center; padding-left: 3rem; } + +section table td:first-child, +section .params td:first-child, +section table thead th:first-child, +section .params thead th:first-child, +section .props thead th:first-child { + border-top-left-radius: .5rem; + border-bottom-left-radius: .5rem; +} + +section table td:last-child, +section .params td:last-child, +section table thead th:last-child, +section .params thead th:last-child, +section .props thead th:last-child { + border-top-right-radius: .5rem; + border-bottom-right-radius: .5rem; +} + +section .params td, +section .params th, +section .props td, +section .props th, +section th, +section td, +section .details .details-item-container { + padding: .5rem 1rem; +} + +section ul, +section ol { + padding-left: 1.5rem; +} + +section ul li, +section ol li { + padding: .25rem 0; +} + +section p { + margin: .75rem 0; +} + +section .details { + border-radius: .5rem; +} + +section .pre-div { + margin: 1rem 0; + border-radius: .5rem; +} + +.param-type { + font-family: "code"; +} + +.signature-attributes { + font-variant: sub; +} + +.method-member-container > strong + dl.param-type > dt { + display: none; +} diff --git a/docs/ucode.js b/docs/ucode.js index 6728cdbd..2377c7ed 100644 --- a/docs/ucode.js +++ b/docs/ucode.js @@ -4,4 +4,36 @@ document.addEventListener('DOMContentLoaded', (ev) => { if (accordionState == null || accordionState == '{}') document.querySelectorAll('[data-isopen="false"]') .forEach(item => item.setAttribute('data-isopen', 'true')); + + const moduleName = location.pathname.match(/\/module-(.+)\.html$/)?.[1]; + + if (moduleName) { + const modulePrefix = `module:${moduleName}.`; + + document.querySelectorAll(`a[href^="module-${CSS.escape(moduleName)}."]`).forEach(a => { + if (a.text?.indexOf(modulePrefix) == 0) + a.text = a.text.substring(modulePrefix.length); + }); + } + + document.querySelectorAll('.param-type, .type-signature').forEach(span => { + let replaced; + do { + replaced = false; + span.innerHTML = span.innerHTML.replace(/\b(Object|Array)\.<((?:(?!&[lg]t;).)+)>/, + (m, t, st) => { + replaced = true; + + if (t == 'Object') + return `Object<${st.replace(/,\s*/, ': ')}>`; + else + return `${st}[]`; + }); + } while (replaced); + }); + + document.querySelectorAll('.type-signature').forEach(span => { + span.innerHTML = span.innerHTML.replace(/\(nullable\) (.+)$/, + '$1nullable'); + }); }); diff --git a/lib.c b/lib.c index d23dde2f..726a74dc 100644 --- a/lib.c +++ b/lib.c @@ -365,7 +365,7 @@ uc_print(uc_vm_t *vm, size_t nargs) * * @param {Object|Array|string} x - The input object, array, or string. * - * @returns {number|null} - The length of the input. + * @returns {?number} - The length of the input. * * @example * length("test") // 4 @@ -484,7 +484,7 @@ uc_index(uc_vm_t *vm, size_t nargs, bool right) * @param {*} needle * The value to find within the array or string. * - * @returns {number|null} + * @returns {?number} * * @example * index("Hello hello hello", "ll") // 2 @@ -516,7 +516,7 @@ uc_lindex(uc_vm_t *vm, size_t nargs) * @param {*} needle * The value to find within the array or string. * - * @returns {number|null} + * @returns {?number} * * @example * rindex("Hello hello hello", "ll") // 14 @@ -844,7 +844,7 @@ uc_exit(uc_vm_t *vm, size_t nargs) * @param {string} [name] * The name of the environment variable. * - * @returns {string|object} + * @returns {string|Object} */ static uc_value_t * uc_getenv(uc_vm_t *vm, size_t nargs) @@ -1037,7 +1037,7 @@ uc_int(uc_vm_t *vm, size_t nargs) * @param {Array} arr * The array to be joined into a string. * - * @returns {string|null} + * @returns {?string} */ static uc_value_t * uc_join(uc_vm_t *vm, size_t nargs) @@ -1073,7 +1073,7 @@ uc_join(uc_vm_t *vm, size_t nargs) * @param {object} obj * The object from which to retrieve the key names. * - * @returns {Array|null} + * @returns {?Array} */ static uc_value_t * uc_keys(uc_vm_t *vm, size_t nargs) @@ -1104,7 +1104,7 @@ uc_keys(uc_vm_t *vm, size_t nargs) * @param {string} s * The input string. * - * @returns {string|null} + * @returns {?string} * The lowercase string. * * @example @@ -1211,7 +1211,7 @@ uc_map(uc_vm_t *vm, size_t nargs) * @param {number} [offset] * The offset of the character. * - * @returns {number|null} + * @returns {?number} * * @example * ord("Abc"); // 65 @@ -1265,7 +1265,7 @@ uc_ord(uc_vm_t *vm, size_t nargs) * @param {*} x * The value to determine the type of. * - * @returns {string|null} + * @returns {?string} */ static uc_value_t * uc_type(uc_vm_t *vm, size_t nargs) @@ -1307,7 +1307,7 @@ uc_type(uc_vm_t *vm, size_t nargs) * @param {Array|string} arr_or_str * The input array or string. * - * @returns {Array|string|null} + * @returns {?(Array|string)} * * @example * reverse([1, 2, 3]); // [ 3, 2, 1 ] @@ -1938,7 +1938,7 @@ uc_time(uc_vm_t *vm, size_t nargs) * @param {*} str * The string to be converted to uppercase. * - * @returns {string|null} + * @returns {?string} * * @example * uc("hello"); // "HELLO" @@ -2040,7 +2040,7 @@ uc_uchr(uc_vm_t *vm, size_t nargs) * @param {*} obj * The object from which to extract values. * - * @returns {Array|null} + * @returns {?Array} * * @example * values({ foo: true, bar: false }); // [true, false] @@ -2774,7 +2774,7 @@ uc_require(uc_vm_t *vm, size_t nargs) * @param {string} address * The IP address string to convert. * - * @returns {number[]|null} + * @returns {?number[]} * * @example * iptoarr("192.168.1.1") // [ 192, 168, 1, 1 ] @@ -2851,7 +2851,7 @@ check_byte(uc_value_t *v) * @param {number[]} arr * The byte array to convert into an IP address string. * - * @returns {string|null} + * @returns {?string} * * @example * arrtoip([ 192, 168, 1, 1 ]) // "192.168.1.1" @@ -2928,7 +2928,7 @@ uc_arrtoip(uc_vm_t *vm, size_t nargs) * @param {RegExp} pattern * The regular expression pattern. * - * @returns {Array|null} + * @returns {?Array} * * @example * match("foobarbaz", /b.(.)/) // ["bar", "r"] @@ -3976,7 +3976,7 @@ uc_trace(uc_vm_t *vm, size_t nargs) * @param {Object} [proto] * The optional prototype object. * - * @returns {Object|null} + * @returns {?Object} * * @example * const arr = [1, 2, 3]; @@ -4222,7 +4222,7 @@ uc_wildcard(uc_vm_t *vm, size_t nargs) * @param {boolean} [dironly] * Whether to return only the directory portion of the source file path. * - * @returns {string|null} + * @returns {?string} * * @example * sourcepath(); // Returns the path of the currently executed file @@ -4426,7 +4426,7 @@ uc_max(uc_vm_t *vm, size_t nargs) * @param {string} str * The base64 encoded string to decode. * - * @returns {string|null} + * @returns {?string} * * @example * b64dec("VGhpcyBpcyBhIHRlc3Q="); // Returns "This is a test" @@ -4576,7 +4576,7 @@ static const char Base64[] = * @param {string} str * The string to encode. * - * @returns {string|null} + * @returns {?string} * * @example * b64enc("This is a test"); // Returns "VGhpcyBpcyBhIHRlc3Q=" @@ -4713,7 +4713,7 @@ uc_uniq_ucv_equal(const void *k1, const void *k2) * @param {Array} array * The input array. * - * @returns {Array|null} + * @returns {?Array} * * @example * uniq([1, true, "foo", 2, true, "bar", "foo"]); // Returns [1, true, "foo", 2, "bar"] @@ -4752,6 +4752,28 @@ uc_uniq(uc_vm_t *vm, size_t nargs) return uniq; } +/** + * A time spec is a plain object describing a point in time, it is returned by + * the {@link module:core#gmtime|gmtime()} and + * {@link module:core#localtime|localtime()} functions and expected as parameter + * by the complementary {@link module:core#timegm|timegm()} and + * {@link module:core#timelocal|timelocal()} functions. + * + * When returned by `gmtime()` or `localtime()`, all members of the object will + * be initialized, when passed as argument to `timegm()` or `timelocal()`, most + * member values are optional. + * + * @typedef {Object} module:core.TimeSpec + * @property {number} sec - Seconds (0..60) + * @property {number} min - Minutes (0..59) + * @property {number} hour - Hours (0..23) + * @property {number} mday - Day of month (1..31) + * @property {number} mon - Month (1..12) + * @property {number} year - Year (>= 1900) + * @property {number} wday - Day of week (1..7, Sunday = 7) + * @property {number} yday - Day of year (1-366, Jan 1st = 1) + * @property {number} isdst - Daylight saving time in effect (yes = 1) + */ static uc_value_t * uc_gettime_common(uc_vm_t *vm, size_t nargs, bool local) { @@ -4782,16 +4804,7 @@ uc_gettime_common(uc_vm_t *vm, size_t nargs, bool local) * containing broken-down date and time information according to the local * system timezone. * - * The resulting dictionary contains the following fields: - * - `sec` Seconds (0-60) - * - `min` Minutes (0-59) - * - `hour` Hours (0-23) - * - `mday` Day of month (1-31) - * - `mon` Month (1-12) - * - `year` Year (>= 1900) - * - `wday` Day of the week (1-7, Sunday = 7) - * - `yday` Day of the year (1-366, Jan 1st = 1) - * - `isdst` Daylight saving time in effect (yes = 1) + * See {@link module:core.TimeSpec|TimeSpec} for a description of the fields. * * Note that in contrast to the underlying `localtime(3)` C library function, * the values for `mon`, `wday`, and `yday` are 1-based, and the `year` is @@ -4802,7 +4815,7 @@ uc_gettime_common(uc_vm_t *vm, size_t nargs, bool local) * @param {number} [epoch] * The epoch timestamp. * - * @returns {Object} + * @returns {module:core.TimeSpec} * * @example * localtime(1647953502); @@ -4828,14 +4841,14 @@ uc_localtime(uc_vm_t *vm, size_t nargs) /** * Like `localtime()` but interpreting the given epoch value as UTC time. * - * See `localtime()` for details on the return value. + * See {@link module:core#localtime|localtime()} for details on the return value. * * @function module:core#gmtime * * @param {number} [epoch] * The epoch timestamp. * - * @returns {Object} + * @returns {module:core.TimeSpec} * * @example * gmtime(1647953502); @@ -4908,9 +4921,9 @@ uc_mktime_common(uc_vm_t *vm, size_t nargs, bool local) } /** - * Performs the inverse operation of `localtime()` by taking a broken-down date - * and time dictionary and transforming it into an epoch value according to the - * local system timezone. + * Performs the inverse operation of {@link module:core#localtime|localtime()} + * by taking a broken-down date and time dictionary and transforming it into an + * epoch value according to the local system timezone. * * The `wday` and `yday` fields of the given date time specification are * ignored. Field values outside of their valid range are internally normalized, @@ -4922,10 +4935,10 @@ uc_mktime_common(uc_vm_t *vm, size_t nargs, bool local) * * @function module:core#timelocal * - * @param {Object} datetimespec + * @param {module:core.TimeSpec} datetimespec * The broken-down date and time dictionary. * - * @returns {number|null} + * @returns {?number} * * @example * timelocal({ "sec": 42, "min": 51, "hour": 13, "mday": 22, "mon": 3, "year": 2022, "isdst": 0 }); @@ -4941,14 +4954,14 @@ uc_timelocal(uc_vm_t *vm, size_t nargs) * Like `timelocal()` but interpreting the given date time specification as UTC * time. * - * See `timelocal()` for details. + * See {@link module:core#timelocal|timelocal()} for details. * * @function module:core#timegm * - * @param {Object} datetimespec + * @param {module:core.TimeSpec} datetimespec * The broken-down date and time dictionary. * - * @returns {number|null} + * @returns {?number} * * @example * timegm({ "sec": 42, "min": 51, "hour": 13, "mday": 22, "mon": 3, "year": 2022, "isdst": 0 }); @@ -4980,7 +4993,7 @@ uc_timegm(uc_vm_t *vm, size_t nargs) * @param {boolean} [monotonic] * Whether to query the monotonic system clock. * - * @returns {Array|null} + * @returns {?number[]} * * @example * clock(); // [ 1647954926, 798269464 ] @@ -5078,7 +5091,7 @@ hexval(unsigned char c, bool lo) * @param {string} [skipchars] * The characters to skip during decoding. * - * @returns {string|null} + * @returns {?string} * * @example * hexdec("48656c6c6f20776f726c64210a"); // "Hello world!\n" @@ -5165,7 +5178,7 @@ uc_hexdec(uc_vm_t *vm, size_t nargs) * @param {*} [argument] * The argument for the operation. * - * @returns {boolean|number|null} + * @returns {?(boolean|number)} * * @example * gc(); // true diff --git a/lib/debug.c b/lib/debug.c index 40b044ec..0b227d9f 100644 --- a/lib/debug.c +++ b/lib/debug.c @@ -667,7 +667,7 @@ debug_setup(uc_vm_t *vm) * @param {string|module:fs.file|module:fs.proc} file * The file path or open file handle to write report to. * - * @return {boolean|null} + * @return {?boolean} */ static uc_value_t * uc_memdump(uc_vm_t *vm, size_t nargs) @@ -818,7 +818,7 @@ uc_traceback(uc_vm_t *vm, size_t nargs) * * @function module:debug#sourcepos * - * @return {module:debug.SourcePosition|null} + * @return {?module:debug.SourcePosition} */ /** @@ -955,7 +955,7 @@ uc_getinfo_upvals(uc_vm_t *vm, uc_closure_t *closure) * @param {*} value * The value to query information for. * - * @return {module:debug.ValueInformation|null} + * @return {?module:debug.ValueInformation} */ /** @@ -1362,7 +1362,7 @@ uc_xlocal(uc_vm_t *vm, uc_value_t *level, uc_value_t *var, uc_value_t **set) * @param {string|number} variable * The variable index or variable name to obtain information for. * - * @returns {module:debug.LocalInfo|null} + * @returns {?module:debug.LocalInfo} */ static uc_value_t * uc_getlocal(uc_vm_t *vm, size_t nargs) @@ -1406,7 +1406,7 @@ uc_getlocal(uc_vm_t *vm, size_t nargs) * @param {*} [value=null] * The value to set the local variable to. * - * @returns {module:debug.LocalInfo|null} + * @returns {?module:debug.LocalInfo} */ static uc_value_t * uc_setlocal(uc_vm_t *vm, size_t nargs) @@ -1552,7 +1552,7 @@ uc_xupval(uc_vm_t *vm, uc_value_t *target, uc_value_t *var, uc_value_t **set) * @param {string|number} variable * The variable index or variable name to obtain information for. * - * @returns {module:debug.UpvalInfo|null} + * @returns {?module:debug.UpvalInfo} */ static uc_value_t * uc_getupval(uc_vm_t *vm, size_t nargs) @@ -1601,7 +1601,7 @@ uc_getupval(uc_vm_t *vm, size_t nargs) * @param {*} value * The value to set the variable to. * - * @returns {module:debug.UpvalInfo|null} + * @returns {?module:debug.UpvalInfo} */ static uc_value_t * uc_setupval(uc_vm_t *vm, size_t nargs) diff --git a/lib/fs.c b/lib/fs.c index 0f03ad4c..4d9695e3 100644 --- a/lib/fs.c +++ b/lib/fs.c @@ -79,7 +79,7 @@ static int last_error = 0; * @function module:fs#error * * - * @returns {string|null} + * @returns {?string} * * @example * // Trigger file system error @@ -258,6 +258,8 @@ uc_fs_fileno_common(uc_vm_t *vm, size_t nargs, const char *type) * @class module:fs.proc * @hideconstructor * + * @borrows module:fs#error as module:fs.proc#error + * * @see {@link module:fs#popen|popen()} * * @example @@ -275,26 +277,6 @@ uc_fs_fileno_common(uc_vm_t *vm, size_t nargs, const char *type) * handle.error(); */ -/** - * Query error information. - * - * Returns a string containing a description of the last occurred error or - * `null` if there is no error information. - * - * @function module:fs.proc#error - * - * @returns {string|null} - * - * @example - * // Trigger error - * const fp = popen("command"); - * fp.close(); - * fp.close(); // already closed - * - * // Print error (should yield "Bad file descriptor") - * print(fp.error(), "\n"); - */ - /** * Closes the program handle and awaits program termination. * @@ -316,7 +298,7 @@ uc_fs_fileno_common(uc_vm_t *vm, size_t nargs, const char *type) * * @function module:fs.proc#close * - * @returns {number|null} + * @returns {?number} */ static uc_value_t * uc_fs_pclose(uc_vm_t *vm, size_t nargs) @@ -378,7 +360,7 @@ uc_fs_pclose(uc_vm_t *vm, size_t nargs) * The length of data to read. Can be a number, the string "line", the string * "all", or a single character string. * - * @returns {string|null} + * @returns {?string} * * @example * const fp = popen("command", "r"); @@ -426,7 +408,7 @@ uc_fs_pread(uc_vm_t *vm, size_t nargs) * @param {*} data * The data to be written. * - * @returns {number|null} + * @returns {?number} * * @example * const fp = popen("command", "w"); @@ -448,7 +430,7 @@ uc_fs_pwrite(uc_vm_t *vm, size_t nargs) * * @function module:fs.proc#flush * - * @returns {boolean|null} + * @returns {?boolean} * */ static uc_value_t * @@ -466,7 +448,7 @@ uc_fs_pflush(uc_vm_t *vm, size_t nargs) * * @function module:fs.proc#fileno * - * @returns {number|null} + * @returns {?number} */ static uc_value_t * uc_fs_pfileno(uc_vm_t *vm, size_t nargs) @@ -498,7 +480,7 @@ uc_fs_pfileno(uc_vm_t *vm, size_t nargs) * @param {string} [mode="r"] * The open mode of the process handle. * - * @returns {module:fs.proc|null} + * @returns {?module:fs.proc} * * @example * // Open a process @@ -531,6 +513,8 @@ uc_fs_popen(uc_vm_t *vm, size_t nargs) * @class module:fs.file * @hideconstructor * + * @borrows module:fs#error as module:fs.file#error + * * @see {@link module:fs#open|open()} * @see {@link module:fs#fdopen|fdopen()} * @see {@link module:fs#mkstemp|mkstemp()} @@ -555,26 +539,6 @@ uc_fs_popen(uc_vm_t *vm, size_t nargs) * handle.error(); */ -/** - * Query error information. - * - * Returns a string containing a description of the last occurred error or - * `null` if there is no error information. - * - * @function module:fs.file#error - * - * @returns {string|null} - * - * @example - * // Trigger error - * const fp = open("file.txt"); - * fp.close(); - * fp.close(); // already closed - * - * // Print error (should yield "Bad file descriptor") - * print(fp.error(), "\n"); - */ - /** * Closes the file handle. * @@ -587,7 +551,7 @@ uc_fs_popen(uc_vm_t *vm, size_t nargs) * * @function module:fs.file#close * - * @returns {boolean|null} + * @returns {?boolean} */ static uc_value_t * uc_fs_close(uc_vm_t *vm, size_t nargs) @@ -639,7 +603,7 @@ uc_fs_close(uc_vm_t *vm, size_t nargs) * The length of data to read. Can be a number, the string "line", the string * "all", or a single character string. * - * @returns {string|null} + * @returns {?string} * * @example * const fp = open("file.txt", "r"); @@ -687,7 +651,7 @@ uc_fs_read(uc_vm_t *vm, size_t nargs) * @param {*} data * The data to be written. * - * @returns {number|null} + * @returns {?number} * * @example * const fp = open("file.txt", "w"); @@ -724,7 +688,7 @@ uc_fs_write(uc_vm_t *vm, size_t nargs) * | `1` | The given offset is relative to the current read position. | * | `2` | The given offset is relative to the end of the file. | * - * @returns {boolean|null} + * @returns {?boolean} * * @example * const fp = open("file.txt", "r"); @@ -785,7 +749,7 @@ uc_fs_seek(uc_vm_t *vm, size_t nargs) * * @function module:fs.file#tell * - * @returns {number|null} + * @returns {?number} */ static uc_value_t * uc_fs_tell(uc_vm_t *vm, size_t nargs) @@ -818,7 +782,7 @@ uc_fs_tell(uc_vm_t *vm, size_t nargs) * * @function module:fs.file#isatty * - * @returns {boolean|null} + * @returns {?boolean} * */ static uc_value_t * @@ -847,7 +811,7 @@ uc_fs_isatty(uc_vm_t *vm, size_t nargs) * * @function module:fs.file#flush * - * @returns {boolean|null} + * @returns {?boolean} * */ static uc_value_t * @@ -865,7 +829,7 @@ uc_fs_flush(uc_vm_t *vm, size_t nargs) * * @function module:fs.file#fileno * - * @returns {number|null} + * @returns {?number} */ static uc_value_t * uc_fs_fileno(uc_vm_t *vm, size_t nargs) @@ -913,7 +877,7 @@ uc_fs_fileno(uc_vm_t *vm, size_t nargs) * @param {number} [perm=0o666] * The file creation permissions (for modes `w…` and `a…`) * - * @returns {module:fs.file|null} + * @returns {?module:fs.file} * * @example * // Open a file in read-only mode @@ -1058,6 +1022,8 @@ uc_fs_fdopen(uc_vm_t *vm, size_t nargs) * @class module:fs.dir * @hideconstructor * + * @borrows module:fs#error as module:fs.dir#error + * * @see {@link module:fs#opendir|opendir()} * * @example @@ -1074,26 +1040,6 @@ uc_fs_fdopen(uc_vm_t *vm, size_t nargs) * handle.error(); */ -/** - * Query error information. - * - * Returns a string containing a description of the last occurred error or - * `null` if there is no error information. - * - * @function module:fs.dir#error - * - * @returns {string|null} - * - * @example - * // Trigger error - * const fp = opendir("/tmp"); - * fp.close(); - * fp.close(); // already closed - * - * // Print error (should yield "Bad file descriptor") - * print(fp.error(), "\n"); - */ - /** * Read the next entry from the open directory. * @@ -1105,7 +1051,7 @@ uc_fs_fdopen(uc_vm_t *vm, size_t nargs) * * @function module:fs.dir#read * - * @returns {string|null} + * @returns {?string} */ static uc_value_t * uc_fs_readdir(uc_vm_t *vm, size_t nargs) @@ -1139,7 +1085,7 @@ uc_fs_readdir(uc_vm_t *vm, size_t nargs) * * @function module:fs.dir#tell * - * @returns {number|null} + * @returns {?number} */ static uc_value_t * uc_fs_telldir(uc_vm_t *vm, size_t nargs) @@ -1174,7 +1120,7 @@ uc_fs_telldir(uc_vm_t *vm, size_t nargs) * @param {number} offset * Position value obtained by `tell()`. * - * @returns {boolean|null} + * @returns {?boolean} * * @example * @@ -1218,7 +1164,7 @@ uc_fs_seekdir(uc_vm_t *vm, size_t nargs) * * @function module:fs.dir#close * - * @returns {boolean|null} + * @returns {?boolean} */ static uc_value_t * uc_fs_closedir(uc_vm_t *vm, size_t nargs) @@ -1247,7 +1193,7 @@ uc_fs_closedir(uc_vm_t *vm, size_t nargs) * @param {string} path * The path to the directory. * - * @returns {module:fs.dir|null} + * @returns {?module:fs.dir} * * @example * // Open a directory @@ -1282,7 +1228,7 @@ uc_fs_opendir(uc_vm_t *vm, size_t nargs) * @param {string} path * The path to the symbolic link. * - * @returns {string|null} + * @returns {?string} * * @example * // Read the value of a symbolic link @@ -1456,7 +1402,7 @@ uc_fs_stat_common(uc_vm_t *vm, size_t nargs, bool use_lstat) * @param {string} path * The path to the file or directory. * - * @returns {module:fs.FileStatResult|null} + * @returns {?module:fs.FileStatResult} * * @example * // Get information about a file @@ -1481,7 +1427,7 @@ uc_fs_stat(uc_vm_t *vm, size_t nargs) * @param {string} path * The path to the file or directory. * - * @returns {module:fs.FileStatResult|null} + * @returns {?module:fs.FileStatResult} * * @example * // Get information about a directory @@ -1505,7 +1451,7 @@ uc_fs_lstat(uc_vm_t *vm, size_t nargs) * @param {string} path * The path to the new directory. * - * @returns {boolean|null} + * @returns {?boolean} * * @example * // Create a directory @@ -1539,7 +1485,7 @@ uc_fs_mkdir(uc_vm_t *vm, size_t nargs) * @param {string} path * The path to the directory to be removed. * - * @returns {boolean|null} + * @returns {?boolean} * * @example * // Remove a directory @@ -1574,7 +1520,7 @@ uc_fs_rmdir(uc_vm_t *vm, size_t nargs) * @param {string} path * The path of the symbolic link. * - * @returns {boolean|null} + * @returns {?boolean} * * @example * // Create a symbolic link @@ -1608,7 +1554,7 @@ uc_fs_symlink(uc_vm_t *vm, size_t nargs) * @param {string} path * The path to the file or symbolic link. * - * @returns {boolean|null} + * @returns {?boolean} * * @example * // Remove a file @@ -1637,7 +1583,7 @@ uc_fs_unlink(uc_vm_t *vm, size_t nargs) * * @function module:fs#getcwd * - * @returns {string|null} + * @returns {?string} * * @example * // Get the current working directory @@ -1692,7 +1638,7 @@ uc_fs_getcwd(uc_vm_t *vm, size_t nargs) * @param {string} path * The path to the new working directory. * - * @returns {boolean|null} + * @returns {?boolean} * * @example * // Change the current working directory @@ -1728,7 +1674,7 @@ uc_fs_chdir(uc_vm_t *vm, size_t nargs) * @param {number} mode * The new mode (permissions). * - * @returns {boolean|null} + * @returns {?boolean} * * @example * // Change the mode of a file @@ -1872,7 +1818,7 @@ uc_fs_resolve_group(uc_value_t *v, gid_t *gid) * The new group's ID. When given as number, it is used as-is, when given as * string, the group name is resolved to the corresponding gid first. * - * @returns {boolean|null} + * @returns {?boolean} * * @example * // Change the owner of a file @@ -1918,7 +1864,7 @@ uc_fs_chown(uc_vm_t *vm, size_t nargs) * @param {string} newPath * The new path of the file or directory. * - * @returns {boolean|null} + * @returns {?boolean} * * @example * // Rename a file @@ -1980,7 +1926,7 @@ uc_fs_glob(uc_vm_t *vm, size_t nargs) * @param {string} path * The path to extract the directory name from. * - * @returns {string|null} + * @returns {?string} * * @example * // Get the directory name of a path @@ -2029,7 +1975,7 @@ uc_fs_dirname(uc_vm_t *vm, size_t nargs) * @param {string} path * The path to extract the base name from. * - * @returns {string|null} + * @returns {?string} * * @example * // Get the base name of a path @@ -2083,7 +2029,7 @@ uc_fs_lsdir_sort_fn(const void *k1, const void *k2) * @param {string} path * The path to the directory. * - * @returns {string[]|null} + * @returns {?string[]} * * @example * // List the content of a directory @@ -2169,7 +2115,7 @@ uc_fs_lsdir(uc_vm_t *vm, size_t nargs) * @param {string} [template="/tmp/XXXXXX"] * The path template to use when forming the temporary file name. * - * @returns {module:fs.file|null} + * @returns {?module:fs.file} * * @example * // Create a unique temporary file in the current working directory @@ -2261,7 +2207,7 @@ uc_fs_mkstemp(uc_vm_t *vm, size_t nargs) * @param {number} [mode="f"] * Optional access mode. * - * @returns {boolean|null} + * @returns {?boolean} * * @example * // Check file read and write accessibility @@ -2329,7 +2275,7 @@ uc_fs_access(uc_vm_t *vm, size_t nargs) * Number of bytes to limit the result to. When omitted, the entire content is * returned. * - * @returns {string|null} + * @returns {?string} * * @example * // Read first 100 bytes of content @@ -2441,7 +2387,7 @@ uc_fs_readfile(uc_vm_t *vm, size_t nargs) * Truncates the amount of data to be written to the specified amount of bytes. * When omitted, the entire content is written. * - * @returns {number|null} + * @returns {?number} * * @example * // Write string to a file @@ -2522,7 +2468,7 @@ uc_fs_writefile(uc_vm_t *vm, size_t nargs) * @param {string} path * The path to the file or directory. * - * @returns {string|null} + * @returns {?string} * * @example * // Resolve the absolute path of a file @@ -2561,7 +2507,7 @@ uc_fs_realpath(uc_vm_t *vm, size_t nargs) * * @function module:fs#pipe * - * @returns {module:fs.file[]|null} + * @returns {?module:fs.file[]} * * @example * // Create a pipe diff --git a/lib/uci.c b/lib/uci.c index b7319bf3..96cce6a0 100644 --- a/lib/uci.c +++ b/lib/uci.c @@ -14,6 +14,40 @@ * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. */ +/** + * # OpenWrt UCI configuration + * + * The `uci` module provides access to the native OpenWrt + * {@link https://github.com/openwrt/uci libuci} API for reading and + * manipulating UCI configuration files. + * + * Functions can be individually imported and directly accessed using the + * {@link https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Statements/import#named_import named import} + * syntax: + * + * ``` + * import { cursor } from 'uci'; + * + * let ctx = cursor(); + * let hostname = ctx.get_first('system', 'system', 'hostname'); + * ``` + * + * Alternatively, the module namespace can be imported + * using a wildcard import statement: + * + * ``` + * import * as uci from 'uci'; + * + * let ctx = uci.cursor(); + * let hostname = ctx.get_first('system', 'system', 'hostname'); + * ``` + * + * Additionally, the uci module namespace may also be imported by invoking + * the `ucode` interpreter with the `-luci` switch. + * + * @module uci + */ + #include #include @@ -30,6 +64,24 @@ enum pkg_cmd { CMD_REVERT }; +/** + * Query error information. + * + * Returns a string containing a description of the last occurred error or + * `null` if there is no error information. + * + * @function module:uci#error + * + * @returns {?string} + * + * @example + * // Trigger error + * const ctx = cursor(); + * ctx.set("not_existing_config", "test", "1"); + * + * // Print error (should yield "Entry not found") + * print(ctx.error(), "\n"); + */ static uc_value_t * uc_uci_error(uc_vm_t *vm, size_t nargs) { @@ -63,6 +115,38 @@ uc_uci_error(uc_vm_t *vm, size_t nargs) } +/** + * Instantiate uci cursor. + * + * A uci cursor is a context for interacting with uci configuration files. It's + * purpose is to cache and hold changes made to loaded configuration states + * until those changes are written out to disk or discared. + * + * Unsaved and uncommitted changes in a cursor instance are private and not + * visible to other cursor instances instantiated by the same program or other + * processes on the system. + * + * Returns the instantiated cursor on success. + * + * Returns `null` on error, e.g. if an invalid path argument was provided. + * + * @function module:uci#cursor + * + * @param {string} [config_dir=/etc/config] + * The directory to search for configuration files. It defaults to the well + * known uci configuration directory `/etc/config` but may be set to a different + * path for special purpose applications. + * + * @param {string} [delta_dir=/tmp/.uci] + * The directory to save delta records in. It defaults to the well known + * `/tmp/.uci` path which is used as default by the uci command line tool. + * + * By changing this path to a different location, it is possible to isolate + * uncommitted application changes from the uci cli or other processes on the + * system. + * + * @returns {?module:uci.cursor} + */ static uc_value_t * uc_uci_cursor(uc_vm_t *vm, size_t nargs) { @@ -98,6 +182,164 @@ uc_uci_cursor(uc_vm_t *vm, size_t nargs) } +/** + * Represents a context for interacting with uci configuration files. + * + * Operations on uci configurations are performed through a uci cursor object + * which operates on in-memory representations of loaded configuration files. + * + * Any changes made to configuration values are local to the cursor object and + * held in memory only until they're written out to the filesystem using the + * `save()` and `commit()` methods. + * + * Changes performed in one cursor instance are not reflected in another, unless + * the first instance writes those changes to the filesystem and the other + * instance explicitly (re)loads the affected configuration files. + * + * @class module:uci.cursor + * @hideconstructor + * + * @borrows module:uci#error as module.uci.cursor#error + * + * @see {@link module:uci#cursor|cursor()} + * + * @example + * + * const ctx = cursor(…); + * + * // Enumerate configuration files + * ctx.configs(); + * + * // Load configuration files + * ctx.load(…); + * ctx.unload(…); + * + * // Query values + * ctx.get(…); + * ctx.get_all(…); + * ctx.get_first(…); + * ctx.foreach(…); + * + * // Modify values + * ctx.add(…); + * ctx.set(…); + * ctx.rename(…); + * ctx.reorder(…); + * ctx.delete(…); + * + * // Stage, revert, save changes + * ctx.changes(…); + * ctx.save(…); + * ctx.revert(…); + * ctx.commit(…); + */ + +/** + * A uci change record is a plain array containing the change operation name as + * first element, the affected section ID as second argument and an optional + * third and fourth argument whose meanings depend on the operation. + * + * @typedef {string[]} ChangeRecord + * @memberof module:uci.cursor + * + * @property {string} 0 + * The operation name - may be one of `add`, `set`, `remove`, `order`, + * `list-add`, `list-del` or `rename`. + * + * @property {string} 1 + * The section ID targeted by the operation. + * + * @property {string} 2 + * The meaning of the third element depends on the operation. + * - For `add` it is type of the section that has been added + * - For `set` it either is the option name if a fourth element exists, or the + * type of a named section which has been added when the change entry only + * contains three elements. + * - For `remove` it contains the name of the option that has been removed. + * - For `order` it specifies the new sort index of the section. + * - For `list-add` it contains the name of the list option a new value has been + * added to. + * - For `list-del` it contains the name of the list option a value has been + * removed from. + * - For `rename` it contains the name of the option that has been renamed if a + * fourth element exists, else it contains the new name a section has been + * renamed to if the change entry only contains three elements. + * + * @property {string} 4 + * The meaning of the fourth element depends on the operation. + * - For `set` it is the value an option has been set to. + * - For `list-add` it is the new value that has been added to a list option. + * - For `rename` it is the new name of an option that has been renamed. + */ + +/** + * A section object represents the options and their corresponding values + * enclosed within a configuration section, as well as some additional meta data + * such as sort indexes and internal ID. + * + * Any internal metadata fields are prefixed with a dot which isn't an allowed + * character for normal option names. + * + * @typedef {Object} SectionObject + * @memberof module:uci.cursor + * + * @property {boolean} .anonymous + * The `.anonymous` property specifies whether the configuration is + * anonymous (`true`) or named (`false`). + * + * @property {number} .index + * The `.index` property specifies the sort order of the section. + * + * @property {string} .name + * The `.name` property holds the name of the section object. It may be either + * an anonymous ID in the form `cfgXXXXXX` with `X` being a hexadecimal digit or + * a string holding the name of the section. + * + * @property {string} .type + * The `.type` property contains the type of the corresponding uci + * section. + * + * @property {string|string[]} * + * A section object may contain an arbitrary number of further properties + * representing the uci option enclosed in the section. + * + * All option property names will be in the form `[A-Za-z0-9_]+` and either + * contain a string value or an array of strings, in case the underlying option + * is an UCI list. + */ + +/** + * The sections callback is invoked for each section found within the given + * configuration and receives the section object and its associated name as + * arguments. + * + * @callback module:uci.cursor.SectionCallback + * + * @param {module:uci.cursor.SectionObject} section + * The section object. + */ + +/** + * Explicitly reload configuration file. + * + * Usually, any attempt to query or modify a value within a given configuration + * will implicitly load the underlying file into memory. By invoking `load()` + * explicitly, a potentially loaded stale configuration is discarded and + * reloaded from the file system, ensuring that the latest state is reflected in + * the cursor. + * + * Returns `true` if the configuration was successfully loaded. + * + * Returns `null` on error, e.g. if the requested configuration does not exist. + * + * @function module:uci.cursor#load + * + * @param {string} config + * The name of the configuration file to load, e.g. `"system"` to load + * `/etc/config/system` into the cursor. + * + * @returns {?boolean} + */ static uc_value_t * uc_uci_load(uc_vm_t *vm, size_t nargs) { @@ -127,6 +369,26 @@ uc_uci_load(uc_vm_t *vm, size_t nargs) return ucv_boolean_new(true); } +/** + * Explicitly unload configuration file. + * + * The `unload()` function forcibly discards a loaded configuration state from + * the cursor so that the next attempt to read or modify that configuration + * will load it anew from the file system. + * + * Returns `true` if the configuration was successfully unloaded. + * + * Returns `false` if the configuration was not loaded to begin with. + * + * Returns `null` on error, e.g. if the requested configuration does not exist. + * + * @function module:uci.cursor#unload + * + * @param {string} config + * The name of the configuration file to unload. + * + * @returns {?boolean} + */ static uc_value_t * uc_uci_unload(uc_vm_t *vm, size_t nargs) { @@ -308,18 +570,139 @@ uc_uci_get_any(uc_vm_t *vm, size_t nargs, bool all) return ucv_string_new(ptr.s->type); } +/** + * Query a single option value or section type. + * + * When invoked with three arguments, the function returns the value of the + * given option, within the specified section of the given configuration. + * + * When invoked with just a config and section argument, the function returns + * the type of the specified section. + * + * In either case, the given configuration is implicitly loaded into the cursor + * if not already present. + * + * Returns the configuration value or section type on success. + * + * Returns `null` on error, e.g. if the requested configuration does not exist + * or if an invalid argument was passed. + * + * @function module:uci.cursor#get + * + * @param {string} config + * The name of the configuration file to query, e.g. `"system"` to query values + * in `/etc/config/system`. + * + * @param {string} section + * The name of the section to query within the configuration. + * + * @param {string} [option] + * The name of the option to query within the section. If omitted, the type of + * the section is returned instead. + * + * @returns {?(string|string[])} + * + * @example + * const ctx = cursor(…); + * + * // Query an option, extended section notation is supported + * ctx.get('system', '@system[0]', 'hostname'); + * + * // Query a section type (should yield 'interface') + * ctx.get('network', 'lan'); + */ static uc_value_t * uc_uci_get(uc_vm_t *vm, size_t nargs) { return uc_uci_get_any(vm, nargs, false); } +/** + * Query a complete section or configuration. + * + * When invoked with two arguments, the function returns all values of the + * specified section within the given configuration as dictionary. + * + * When invoked with just a config argument, the function returns a nested + * dictionary of all sections present within the given configuration. + * + * In either case, the given configuration is implicitly loaded into the cursor + * if not already present. + * + * Returns the section or configuration dictionary on success. + * + * Returns `null` on error, e.g. if the requested configuration does not exist + * or if an invalid argument was passed. + * + * @function module:uci.cursor#get_all + * + * @param {string} config + * The name of the configuration file to query, e.g. `"system"` to query values + * in `/etc/config/system`. + * + * @param {string} [section] + * The name of the section to query within the configuration. If omitted a + * nested dictionary containing all section values is returned. + * + * @returns {?(Object|module:uci.cursor.SectionObject)} + * + * @example + * const ctx = cursor(…); + * + * // Query all lan interface details + * ctx.get_all('network', 'lan'); + * + * // Dump the entire dhcp configuration + * ctx.get_all('dhcp'); + */ static uc_value_t * uc_uci_get_all(uc_vm_t *vm, size_t nargs) { return uc_uci_get_any(vm, nargs, true); } +/** + * Query option value or name of first section of given type. + * + * When invoked with three arguments, the function returns the value of the + * given option within the first found section of the specified type in the + * given configuration. + * + * When invoked with just a config and section type argument, the function + * returns the name of the first found section of the given type. + * + * In either case, the given configuration is implicitly loaded into the cursor + * if not already present. + * + * Returns the configuration value or section name on success. + * + * Returns `null` on error, e.g. if the requested configuration does not exist + * or if an invalid argument was passed. + * + * @function module:uci.cursor#get_first + * + * @param {string} config + * The name of the configuration file to query, e.g. `"system"` to query values + * in `/etc/config/system`. + * + * @param {string} type + * The section type to find the first section for within the configuration. + * + * @param {string} [option] + * The name of the option to query within the section. If omitted, the name of + * the section is returned instead. + * + * @returns {?(string|string[])} + * + * @example + * const ctx = cursor(…); + * + * // Query hostname in first anonymous "system" section of /etc/config/system + * ctx.get_first('system', 'system', 'hostname'); + * + * // Figure out name of first network interface section (usually "loopback") + * ctx.get_first('network', 'interface'); + */ static uc_value_t * uc_uci_get_first(uc_vm_t *vm, size_t nargs) { @@ -378,6 +761,49 @@ uc_uci_get_first(uc_vm_t *vm, size_t nargs) err_return(UCI_ERR_NOTFOUND); } +/** + * Add anonymous section to given configuration. + * + * Adds a new anonymous (unnamed) section of the specified type to the given + * configuration. In order to add a named section, the three argument form of + * `set()` should be used instead. + * + * In contrast to other query functions, `add()` will not implicitly load the + * configuration into the cursor. The configuration either needs to be loaded + * explicitly through `load()` beforehand, or implicitly by querying it through + * one of the `get()`, `get_all()`, `get_first()` or `foreach()` functions. + * + * Returns the autogenerated, ephemeral name of the added unnamed section + * on success. + * + * Returns `null` on error, e.g. if the targeted configuration was not loaded or + * if an invalid section type value was passed. + * + * @function module:uci.cursor#add + * + * @param {string} config + * The name of the configuration file to add the section to, e.g. `"system"` to + * modify `/etc/config/system`. + * + * @param {string} type + * The type value to use for the added section. + * + * @returns {?string} + * + * @example + * const ctx = cursor(…); + * + * // Load firewall configuration + * ctx.load('firewall'); + * + * // Add unnamed `config rule` section + * const sid = ctx.add('firewall', 'rule'); + * + * // Set values on the newly added section + * ctx.set('firewall', sid, 'name', 'A test'); + * ctx.set('firewall', sid, 'target', 'ACCEPT'); + * … + */ static uc_value_t * uc_uci_add(uc_vm_t *vm, size_t nargs) { @@ -458,6 +884,60 @@ uval_to_uci(uc_vm_t *vm, uc_value_t *val, const char **p, bool *is_list) } } +/** + * Set option value or add named section in given configuration. + * + * When invoked with four arguments, the function sets the value of the given + * option within the specified section of the given configuration to the + * provided value. A value of `""` (empty string) can be used to delete an + * existing option. + * + * When invoked with three arguments, the function adds a new named section to + * the given configuration, using the specified type. + * + * In either case, the given configuration is implicitly loaded into the cursor + * if not already present. + * + * Returns the `true` if the named section was added or the specified option was + * set. + * + * Returns `null` on error, e.g. if the targeted configuration was not found or + * if an invalid value was passed. + * + * @function module:uci.cursor#set + * + * @param {string} config + * The name of the configuration file to set values in, e.g. `"system"` to + * modify `/etc/config/system`. + * + * @param {string} section + * The section name to create or set a value in. + * + * @param {string} option_or_type + * The option name to set within the section or, when the subsequent value + * argument is omitted, the type of the section to create within the + * configuration. + * + * @param {(Array|string|boolean|number)} [value] + * The option value to set. + * + * @returns {?boolean} + * + * @example + * const ctx = cursor(…); + * + * // Add named `config interface guest` section + * ctx.set('network', 'guest', 'interface'); + * + * // Set values on the newly added section + * ctx.set('network', 'guest', 'proto', 'static'); + * ctx.set('network', 'guest', 'ipaddr', '10.0.0.1/24'); + * ctx.set('network', 'guest', 'dns', ['8.8.4.4', '8.8.8.8']); + * … + * + * // Delete 'disabled' option in first wifi-iface section + * ctx.set('wireless', '@wifi-iface[0]', 'disabled', ''); + */ static uc_value_t * uc_uci_set(uc_vm_t *vm, size_t nargs) { @@ -563,6 +1043,50 @@ uc_uci_set(uc_vm_t *vm, size_t nargs) return ucv_boolean_new(true); } +/** + * Delete an option or section from given configuration. + * + * When invoked with three arguments, the function deletes the given option + * within the specified section of the given configuration. + * + * When invoked with two arguments, the function deletes the entire specified + * section within the given configuration. + * + * In either case, the given configuration is implicitly loaded into the cursor + * if not already present. + * + * Returns the `true` if specified option or section has been deleted. + * + * Returns `null` on error, e.g. if the targeted configuration was not found or + * if an invalid value was passed. + * + * @function module:uci.cursor#delete + * + * @param {string} config + * The name of the configuration file to delete values in, e.g. `"system"` to + * modify `/etc/config/system`. + * + * @param {string} section + * The section name to remove the specified option in or, when the subsequent + * argument is omitted, the section to remove entirely. + * + * @param {string} [option] + * The option name to remove within the section. + * + * @returns {?boolean} + * + * @example + * const ctx = cursor(…); + * + * // Delete 'disabled' option in first wifi-iface section + * ctx.delete('wireless', '@wifi-iface[0]', 'disabled'); + * + * // Delete 'wan' interface + * ctx.delete('network', 'lan'); + * + * // Delete last firewall rule + * ctx.delete('firewall', '@rule[-1]'); + */ static uc_value_t * uc_uci_delete(uc_vm_t *vm, size_t nargs) { @@ -598,6 +1122,55 @@ uc_uci_delete(uc_vm_t *vm, size_t nargs) return ucv_boolean_new(true); } +/** + * Rename an option or section in given configuration. + * + * When invoked with four arguments, the function renames the given option + * within the specified section of the given configuration to the provided + * value. + * + * When invoked with three arguments, the function renames the entire specified + * section to the provided value. + * + * In either case, the given configuration is implicitly loaded into the cursor + * if not already present. + * + * Returns the `true` if specified option or section has been renamed. + * + * Returns `null` on error, e.g. if the targeted configuration was not found or + * if an invalid value was passed. + * + * @function module:uci.cursor#rename + * + * @param {string} config + * The name of the configuration file to rename values in, e.g. `"system"` to + * modify `/etc/config/system`. + * + * @param {string} section + * The section name to rename or to rename an option in. + * + * @param {string} option_or_name + * The option name to rename within the section or, when the subsequent name + * argument is omitted, the new name of the renamed section within the + * configuration. + * + * @param {string} [name] + * The new name of the option to rename. + * + * @returns {?boolean} + * + * @example + * const ctx = cursor(…); + * + * // Assign explicit name to last anonymous firewall rule section + * ctx.rename('firewall', '@rule[-1]', 'my_block_rule'); + * + * // Rename 'server' to 'orig_server_list' in ntp section of system config + * ctx.rename('system', 'ntp', 'server', 'orig_server_list'); + * + * // Rename 'wan' interface to 'external' + * ctx.rename('network', 'wan', 'external'); + */ static uc_value_t * uc_uci_rename(uc_vm_t *vm, size_t nargs) { @@ -658,6 +1231,74 @@ uc_uci_rename(uc_vm_t *vm, size_t nargs) return ucv_boolean_new(true); } +/** + * Reorder sections in given configuration. + * + * The `reorder()` function moves a single section by repositioning it to the + * given index within the configurations section list. + * + * The given configuration is implicitly loaded into the cursor if not already + * present. + * + * Returns the `true` if specified section has been moved. + * + * Returns `null` on error, e.g. if the targeted configuration was not found or + * if an invalid value was passed. + * + * @function module:uci.cursor#reorder + * + * @param {string} config + * The name of the configuration file to move the section in, e.g. `"system"` to + * modify `/etc/config/system`. + * + * @param {string} section + * The section name to move. + * + * @param {number} index + * The target index to move the section to, starting from `0`. + * + * @param {string} [name] + * The new name of the option to rename. + * + * @returns {?boolean} + * + * @example + * const ctx = cursor(…); + * + * // Query whole firewall config and reorder resulting dict by type and name + * const type_order = ['defaults', 'zone', 'forwarding', 'redirect', 'rule']; + * const values = ctx.get_all('firewall'); + * + * sort(values, (k1, k2, s1, s2) => { + * // Get weight from type_order array + * let w1 = index(type_order, s1['.type']); + * let w2 = index(type_order, s2['.type']); + * + * // For unknown type orders, use type value itself as weight + * if (w1 == -1) w1 = s1['.type']; + * if (w2 == -1) w2 = s2['.type']; + * + * // Get name from name option, fallback to section name + * let n1 = s1.name ?? k1; + * let n2 = s2.name ?? k2; + * + * // Order by weight + * if (w1 < w2) return -1; + * if (w1 > w2) return 1; + * + * // For same weight order by name + * if (n1 < n2) return -1; + * if (n1 > n2) return 1; + * + * return 0; + * }); + * + * // Sequentially reorder sorted sections in firewall configuration + * let position = 0; + * + * for (let sid in values) + * ctx.reorder('firewall', sid, position++); + */ static uc_value_t * uc_uci_reorder(uc_vm_t *vm, size_t nargs) { @@ -766,18 +1407,125 @@ uc_uci_pkg_command(uc_vm_t *vm, size_t nargs, enum pkg_cmd cmd) return ucv_boolean_new(true); } +/** + * Save accumulated cursor changes to delta directory. + * + * The `save()` function writes consolidated changes made to in-memory copies of + * loaded configuration files to the uci delta directory which effectively makes + * them available to other processes using the same delta directory path as well + * as the `uci changes` cli command when using the default delta directory. + * + * Note that uci deltas are overlayed over the actual configuration file values + * so they're reflected by `get()`, `foreach()` etc. even if the underlying + * configuration files are not actually changed (yet). The delta records may be + * either permanently merged into the configuration by invoking `commit()` or + * reverted through `revert()` in order to restore the current state of the + * underlying configuration file. + * + * When the optional "config" parameter is omitted, delta records for all + * currently loaded configuration files are written. + * + * In case that neither sharing changes with other processes nor any revert + * functionality is required, changes may be committed directly using `commit()` + * instead, bypassing any delta record creation. + * + * Returns the `true` if operation completed successfully. + * + * Returns `null` on error, e.g. if the requested configuration was not loaded + * or when a file system error occurred. + * + * @function module:uci.cursor#save + * + * @param {string} [config] + * The name of the configuration file to save delta records for, e.g. `"system"` + * to store changes for `/etc/config/system`. + * + * @returns {?boolean} + * + * @example + * const ctx = cursor(…); + * + * ctx.set('wireless', '@wifi-iface[0]', 'disabled', '1'); + * ctx.save('wireless'); + * + * @see {@link module:uci.cursor#commit|commit()} + * @see {@link module:uci.cursor#revert|revert()} + */ static uc_value_t * uc_uci_save(uc_vm_t *vm, size_t nargs) { return uc_uci_pkg_command(vm, nargs, CMD_SAVE); } +/** + * Update configuration files with accumulated cursor changes. + * + * The `commit()` function merges changes made to in-memory copies of loaded + * configuration files as well as existing delta records in the cursors + * configured delta directory and writes them back into the underlying + * configuration files, persistently committing changes to the file system. + * + * When the optional "config" parameter is omitted, all currently loaded + * configuration files with either present delta records or yet unsaved + * cursor changes are updated. + * + * Returns the `true` if operation completed successfully. + * + * Returns `null` on error, e.g. if the requested configuration was not loaded + * or when a file system error occurred. + * + * @function module:uci.cursor#commit + * + * @param {string} [config] + * The name of the configuration file to commit, e.g. `"system"` to update the + * `/etc/config/system` file. + * + * @returns {?boolean} + * + * @example + * const ctx = cursor(…); + * + * ctx.set('system', '@system[0]', 'hostname', 'example.org'); + * ctx.commit('system'); + */ static uc_value_t * uc_uci_commit(uc_vm_t *vm, size_t nargs) { return uc_uci_pkg_command(vm, nargs, CMD_COMMIT); } +/** + * Revert accumulated cursor changes and associated delta records. + * + * The `revert()` function discards any changes made to in-memory copies of + * loaded configuration files and discards any related existing delta records in + * the cursors configured delta directory. + * + * When the optional "config" parameter is omitted, all currently loaded + * configuration files with either present delta records or yet unsaved + * cursor changes are reverted. + * + * Returns the `true` if operation completed successfully. + * + * Returns `null` on error, e.g. if the requested configuration was not loaded + * or when a file system error occurred. + * + * @function module:uci.cursor#revert + * + * @param {string} [config] + * The name of the configuration file to revert, e.g. `"system"` to discard any + * changes for the `/etc/config/system` file. + * + * @returns {?boolean} + * + * @example + * const ctx = cursor(…); + * + * ctx.set('system', '@system[0]', 'hostname', 'example.org'); + * ctx.revert('system'); + * + * @see {@link module:uci.cursor#save|save()} + */ static uc_value_t * uc_uci_revert(uc_vm_t *vm, size_t nargs) { @@ -873,6 +1621,38 @@ changes_to_uval(uc_vm_t *vm, struct uci_context *ctx, const char *package) return a; } +/** + * Enumerate pending changes. + * + * The `changes()` function returns a list of change records for currently + * loaded configuration files, originating both from the cursors associated + * delta directory and yet unsaved cursor changes. + * + * When the optional "config" parameter is specified, the requested + * configuration is implicitly loaded if it not already loaded into the cursor. + * + * Returns a dictionary of change record arrays, keyed by configuration name. + * + * Returns `null` on error, e.g. if the requested configuration could not be + * loaded. + * + * @function module:uci.cursor#changes + * + * @param {string} [config] + * The name of the configuration file to enumerate changes for, e.g. `"system"` + * to query pending changes for the `/etc/config/system` file. + * + * @returns {?Object} + * + * @example + * const ctx = cursor(…); + * + * // Enumerate changes for all currently loaded configurations + * const deltas = ctx.changes(); + * + * // Explicitly load and enumerate changes for the "system" configuration + * const deltas = ctx.changes('system'); + */ static uc_value_t * uc_uci_changes(uc_vm_t *vm, size_t nargs) { @@ -907,6 +1687,48 @@ uc_uci_changes(uc_vm_t *vm, size_t nargs) return res; } +/** + * Iterate configuration sections. + * + * The `foreach()` function iterates all sections of the given configuration, + * optionally filtered by type, and invokes the given callback function for + * each encountered section. + * + * When the optional "type" parameter is specified, the callback is only invoked + * for sections of the given type, otherwise it is invoked for all sections. + * + * The requested configuration is implicitly loaded into the cursor. + * + * Returns `true` if the callback was executed successfully at least once. + * + * Returns `false` if the callback was never invoked, e.g. when the + * configuration is empty or contains no sections of the given type. + * + * Returns `null` on error, e.g. when an invalid callback was passed or the + * requested configuration not found. + * + * @function module:uci.cursor#foreach + * + * @param {string} config + * The configuration to iterate sections for, e.g. `"system"` to read the + * `/etc/config/system` file. + * + * @param {?string} type + * Invoke the callback only for sections of the specified type. + * + * @param {module:uci.cursor.SectionCallback} callback + * The callback to invoke for each section, will receive a section dictionary + * as sole argument. + * + * @returns {?boolean} + * + * @example + * const ctx = cursor(…); + * + * // Iterate all network interfaces + * ctx.foreach('network', 'interface', + * section => print(`Have interface ${section[".name"]}\n`)); + */ static uc_value_t * uc_uci_foreach(uc_vm_t *vm, size_t nargs) { @@ -967,6 +1789,26 @@ uc_uci_foreach(uc_vm_t *vm, size_t nargs) return ucv_boolean_new(ret); } +/** + * Enumerate existing configurations. + * + * The `configs()` function yields an array of configuration files present in + * the cursors associated configuration directory, `/etc/config/` by default. + * + * Returns an array of configuration names on success. + * + * Returns `null` on error, e.g. due to filesystem errors. + * + * @function module:uci.cursor#configs + * + * @returns {?string[]} + * + * @example + * const ctx = cursor(…); + * + * // Enumerate all present configuration file names + * const configurations = ctx.configs(); + */ static uc_value_t * uc_uci_configs(uc_vm_t *vm, size_t nargs) {