Chinaunix首页 | 论坛 | 博客
  • 博客访问: 3167309
  • 博文数量: 685
  • 博客积分: 0
  • 博客等级: 民兵
  • 技术积分: 5303
  • 用 户 组: 普通用户
  • 注册时间: 2014-04-19 14:17
个人简介

文章分类

全部博文(685)

文章存档

2015年(116)

2014年(569)

分类: 嵌入式

2014-11-07 16:40:48

原文地址:http://blog.csdn.net/jk110333/article/details/19479747

OpenWRT UCI API的使用

UCI 是OpenWRT为实现配置集中化而引入的一个软件包, 通过修改UCI,可以实现对OpenWRT的绝对部分配置的修改.LUCI(OpenWRT 的WEB配置界面)也是通过读UCI配置文件的操作来实现用户对路由的配置的。通过掌握UCI的API的使用,可以方便地将您的软件的配置接口集成到LUCI中.


LUCI配置文件简介



LUCI的配置文件一般存储在 /etc/config目录下。比如网络配置文件则是 /etc/config/network 无线的配置文件是 /etc/config/wireless. 跟多配置文件的含义参考官方 WIKI


基本概念



UCI上下文: struct uci_context *
包(Package): 一个包对应一个UCI格式的文件.类型是 struct uci_package *
节(Section): 一个配置文件的节点. 类型是 struct uci_list *
值(Value):一个节下面可能包含多个值 一个值具有一个名字.
UCI配置文件的基本操作.


首先您需要引入头文件



  1. #include
  2. #include
  3. #include
  4. #include
  5. static struct uci_context * ctx = NULL; //定义一个UCI上下文的静态变量.
  6. /*********************************************
  7. * 载入配置文件,并遍历Section.
  8. */
  9. bool load_config()
  10. {
  11. struct uci_package * pkg = NULL;
  12. struct uci_element *e;
  13. ctx = uci_alloc_context(); // 申请一个UCI上下文.
  14. if (UCI_OK != uci_load(ctx, UCI_CONFIG_FILE, &pkg))
  15. goto cleanup; //如果打开UCI文件失败,则跳到末尾 清理 UCI 上下文.
  16. /*遍历UCI的每一个节*/
  17. uci_foreach_element(&pkg->sections, e)
  18. {
  19. struct uci_section *s = uci_to_section(e);
  20. // 将一个 element 转换为 section类型, 如果节点有名字,则 s->anonymous 为 false.
  21. // 此时通过 s->e->name 来获取.
  22. // 此时 您可以通过 uci_lookup_option()来获取 当前节下的一个值.
  23. if (NULL != (value = uci_lookup_option_string(ctx, s, "ipaddr")))
  24. {
  25. ip = strdup(value) //如果您想持有该变量值,一定要拷贝一份。当 pkg销毁后value的内存会被释放。
  26. }
  27. // 如果您不确定是 string类型 可以先使用 uci_lookup_option() 函数得到Option 然后再判断.
  28. // Option 的类型有 UCI_TYPE_STRING 和 UCI_TYPE_LIST 两种.
  29. }
  30. uci_unload(ctx, pkg); // 释放 pkg
  31. cleanup:
  32. uci_free_context(ctx);
  33. ctx = NULL;
  34. }


遍历一个UCI_TYPE_LIST 类型.



加入现在有一个如下的配置文件:


  1. config "server" "webserver"
  2. list "index" "index.html"
  3. list "index" "index.php"
  4. list "index" "default.html"


代码片:


  1. // s 为 section.
  2. struct uci_option * o = uci_lookup_option(ctx, s, "index");
  3. if ((NULL != o) && (UCI_TYPE_LIST == o->type)) //o存在 且 类型是 UCI_TYPE_LIST则可以继续.
  4. {
  5. struct uci_element *e;
  6. uci_foreach_element(&o->v.list, e)
  7. {
  8. //这里会循环遍历 list
  9. // e->name 的值依次是 index.html, index.php, default.html
  10. }
  11. }


写配置



UCI提供了一个简洁的办法来操作配置信息,例如有一个配置文件


  1. #文件名: testconfig
  2. config 'servver'
  3. option 'value' '123' # 我们想修改 'value' 的值为 '456'


代码如下:


  1. struct uci_context * ctx = uci_alloc_context(); //申请上下文
  2. struct uci_ptr ptr ={
  3. .package = "config",
  4. .section = "servver",
  5. .option = "value",
  6. .value = "256",
  7. };
  8. uci_set(_ctx,&ptr); //写入配置
  9. uci_commit(_ctx, &ptr.p, false); //提交保存更改
  10. uci_unload(_ctx,ptr.p); //卸载包
  11. uci_free_context(ctx); //释放上下文


依照上面的例子,我们可以举一反三, uci_ptr 用来指定信息.而是用uci_set则是写入信息.同类的函数有如下几个: 针对list的操作:
  1. uci_add_list() // 添加一个list 值
  2. uci_del_list() // 删除一个list 值
  3. uci_delete() // 删除一个option值

阅读(1685) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~