1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245
|
# dbusutil
[包文档](https://godoc.org/github.com/linuxdeepin/go-lib/dbusutil)
## 一般用法
首先创建 Service。
用 NewSessionService 创建 session 服务; 用 NewSystemService 创建 system 服务。
```go
service := dbusutil.NewSessionServer()
```
然后导出对象
用 Export 。
```go
e := &Exportable1{}
service.Export(e)
```
e 要实现 dbusutil.Exportable 接口,并且必须类型必须是 struct 指针。
```
func (e *Exportable1) GetDBusExportInfo() dbusutil.ExportInfo {
return dbusutil.ExportInfo{
Path: "/the/object/path"
Interface: "the.interface.Name"
}
}
```
之后申请服务名,调用 service 的 RequestName 方法, 如果服务名已被其他连接申请了,就会返回错误。
```
service.RequestName("the.service.Name")
```
## 临时服务
服务在运行一段时间内没有接到其他调用请求,就去自动退出。
```
service.SetAutoQuitHanlder(1*time.Minute, func() bool {
// 这个函数叫 canQuit 回调
if canQuit {
// 满足条件退出
return true
}
// 不满足条件,不退出
return false
})
// 等待退出信号
service.Wait()
```
由于没有修改 dbus 库,所有必须在每个导出方法的实现内加上调用 Service.DelayAutoQuit 的方法,如:
```
type Exportable1 struct{
service *dbusutil.Service
}
func (e Exportable1) Method1() *dbus.Error {
e.service.DelayAutoQuit()
return nil
}
```
这样,每次有其他程序通过 dbus 调用了 Method1,就会延长本服务的存活时间。
目前的实现是 2 倍于 SetAutoQuitHandler 第一个参数的值的时间段内没有接到调用请求就会退出。
## 自省
dbusutil 包使用了 go 语言的反射机制自动生成导出对象的 introspection xml。
### 方法
由于反射机制不能获取方法的参数名,只能通过添加额外信息的方式提供方法的参数名,要求必须提供,否则就崩溃。
在结构体里面增加字段,名为 methods, 类型为结构体指针,具体字段立即定义,无需命名结构。
```
type Exportable1 {
methods *struct{
Method1 func() `in:"a,b" out:"result"`
Method2 func() `in:"a,b"`
Method3 func() `out:"result"`
}
}
```
结构中每一个字段的名与导出的方法名对应,类型可为任意类型,但一般写成 func() 即函数指针,字段可以使用 in、out tag, in tag 内容为传入参数名列表,out tag 内容为返回参数名列表,参数名列表用逗号(,) 分割。 注意 in tag 和 out tag 之间要用空格分割。
如果导出方法没有输入和输出参数,则可以省略掉该字段。
### 信号
在结构图里面增加字段,名为 signals, 类型为结构体指针,具体字段立即定义,无需命名结构。
```
type Exportable1 struct {
signals *struct{
Signal1 struct{}
Signal2 struct{
name string
}
Signal3 struct{
name string
value uint32
}
}
}
```
这个匿名结构的每个字段,表示一个信号,类型必须为 struct, 这个更内层的 struct 内每个字段表示信号的参数。
为什么不用 func() 表示信号,是因为 go 的反射机制是不能获取方法的参数名,只能获取参数类型,虽然 d-feet 不能显示出信号的每个参数名是什么,但 D-Bus Introspection 规范是允许的。
### 属性
结构体内每个大写字母开头的字段只要同时满足如下所有条件都可以成为属性。
0. 不是 PropsMaster
1. 不是前一个字段的锁字段
2. 不带 prop tag 不为 "-"
3. 不带空结构,空结构在 dbus 中是非法的。
比如:
```
type Exportable1 struct{
PropsMaster dbusutil.PropsMaster
Prop1 string
Prop1Mu sync.RWMutex
Prop2 uint32
Prop3 struct{}
Prop4 string `prop:"-"`
}
```
最后导出的属性只有 Prop1 和 Prop2。(目前还有 Prop3,但是不能用)
* PropsMaster 是一个特殊的字段
* Prop1Mu 是 Prop1 的锁字段,它紧跟 Prop1 字段,并且名称是前一个字段名 Prop1 + Mu,并且类型是 sync.RWMutex。
* Prop3 的类型是空结构
* Prop4 使用了 prop tag 来明确忽略它。
属性字段可以使用 prop tag, 内容的格式是:
```
选项1:选项值1,选项2:选项值2
```
如 access:rw,emit:false
支持的选项有 access 和 emit。
access 选项值可以为 r, read, w, write, rw, readwrite 之一,默认为 read。
* r 或 read 表示 只读
* w 或 write 表示 只写
* rw 或 readwrite 表示 可读可写
emit 选项值可以为 true, false, invaliates 之一,默认为 true。
* true 表示属性被修改后,发送属性改变信号时带上改变的具体值;
* false 表示属性被修改后,不发送属性改变信号;
* invalidates 表示属性被修改后,发送属性改变信号是不带上具体的值;
#### 属性锁
org.freedestkop.DBus.Properties interface 下的 Get、GetAll 和Set 方法如果有可用的锁,那么就都会自动加锁。Get 与 GetAll 加读锁, Set 加写锁。
目前有三种锁:
全局锁 PropsMaster dbusutil.PropsMaster, 用于值类型的字段,比如 uint32,uint64, string 类型,没有局部锁或内部锁的属性都会使用全局锁。
局部锁 PropXXXMu sync.RWMutex, 用于引用类型的字段,比如 map 类型;要求局部锁字段的名为前一个字段名加 Mu,类型为 sync.RWMutex,作为前一个字段的局部锁。
内部锁, 实现了 dbusutil.Property 接口类型的字段,比如
dbusutil/gsprop 包内的各种类型, 要求 gsprop 包内部实现加锁。
```
type Exportable1 struct {
PropsMaster dbusutil.PropsMaster
Prop1 uint32
Prop2 uint32
Prop3 map[string]string
Prop3Mu sync.RWMutex
Prop4 *gsprop.String
}
```
Prop1 与 Prop2 使用 PropsMaster 加锁
Prop3 使用 Prop3Mu 加锁
Prop4 使用 gsprop.String 内部的锁,在 dbusutil 代码里就是不加锁。
#### dbusutil-gen
利用 go 语言的 generate 特性自动生成属性的 set 与 get 方法,比如操作属性 Name 的 setPropName 与 getPropName 方法,get 方法内实现了加读锁, set 方法内部实现了加写锁与发送改变信号。
先安装 dbusutil-gen 到 $GOPATH/bin 中,执行命令:
```
go install pkg.deepin.io/lib/dbusutil/_tool/dbusutil-gen
```
在代码里写上特殊注释
```
//go:generate dbusutil-gen -type Type1,Type2 file1.go file2.go
```
Type1,Type2 是在 dbus 上导出的类型,用逗号分割。
file1.go file2.go 是要扫描的 go 源代码文件, 其中就定义了 Type1 和 Type2。
然后进入go源码文件所在目录,执行命令 `go generate`, 将看到 dbusutil-gen 的输出信息,自动生成文件 包名 + _dbusutil.go, 如包 pkg1 生成文件名为 pkg1_dbusutil.go, 还可以用 -output 选项指定。
具体选项查看 dbusutil-gen 的帮助信息
可导出属性的字段的注释第一行可以写上
```
// dbusutil-gen: XXXX
```
作为 dbusutil-gen 工具的指导指令
支持的指导指令有
* ignore 忽略此字段
* ignore-below 忽略之后的所有字段
* equal=EQUAL 用于比较新旧值
EQUAL 可以为:
* nil 不进行比较
* 任意,如 bytes.Equal,作为函数,比较是使用 bytes.Equal(old, new)
* 以method: 开头,如 method:equal, 作为方法,比较时使用 old.equal(new)
|