conf.go 9.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428
  1. // conf.go - Functions for pacman.conf parsing.
  2. //
  3. // Copyright (c) 2013 The go-alpm Authors
  4. //
  5. // MIT Licensed. See LICENSE for details.
  6. package alpm
  7. import (
  8. "bufio"
  9. "bytes"
  10. "fmt"
  11. "io"
  12. "os"
  13. "reflect"
  14. "strconv"
  15. "strings"
  16. "syscall"
  17. )
  18. type PacmanOption uint
  19. const (
  20. ConfUseSyslog PacmanOption = 1 << iota
  21. ConfColor
  22. ConfTotalDownload
  23. ConfCheckSpace
  24. ConfVerbosePkgLists
  25. ConfILoveCandy
  26. )
  27. var optionsMap = map[string]PacmanOption{
  28. "UseSyslog": ConfUseSyslog,
  29. "Color": ConfColor,
  30. "TotalDownload": ConfTotalDownload,
  31. "CheckSpace": ConfCheckSpace,
  32. "VerbosePkgLists": ConfVerbosePkgLists,
  33. "ILoveCandy": ConfILoveCandy,
  34. }
  35. // PacmanConfig is a type for holding pacman options parsed from pacman
  36. // configuration data passed to ParseConfig.
  37. type PacmanConfig struct {
  38. RootDir string
  39. DBPath string
  40. CacheDir []string
  41. HookDir []string
  42. GPGDir string
  43. LogFile string
  44. HoldPkg []string
  45. IgnorePkg []string
  46. IgnoreGroup []string
  47. Include []string
  48. Architecture string
  49. XferCommand string
  50. NoUpgrade []string
  51. NoExtract []string
  52. CleanMethod []string
  53. SigLevel SigLevel
  54. LocalFileSigLevel SigLevel
  55. RemoteFileSigLevel SigLevel
  56. UseDelta float64
  57. Options PacmanOption
  58. Repos []RepoConfig
  59. }
  60. // RepoConfig is a type that stores the signature level of a repository
  61. // specified in the pacman config file.
  62. type RepoConfig struct {
  63. Name string
  64. SigLevel SigLevel
  65. Usage Usage
  66. Servers []string
  67. }
  68. // Constants for pacman configuration parsing
  69. const (
  70. tokenSection = iota
  71. tokenKey
  72. tokenComment
  73. )
  74. type iniToken struct {
  75. Type uint
  76. Name string
  77. Values []string
  78. }
  79. type confReader struct {
  80. *bufio.Reader
  81. Lineno uint
  82. }
  83. // newConfReader reads from the io.Reader if it is buffered and returns a
  84. // confReader containing the number of bytes read and 0 for the first line. If
  85. // r is not a buffered reader, a new buffered reader is created using r as its
  86. // input and returned.
  87. func newConfReader(r io.Reader) confReader {
  88. if buf, ok := r.(*bufio.Reader); ok {
  89. return confReader{buf, 0}
  90. }
  91. buf := bufio.NewReader(r)
  92. return confReader{buf, 0}
  93. }
  94. func (rdr *confReader) ParseLine() (tok iniToken, err error) {
  95. line, overflow, err := rdr.ReadLine()
  96. switch {
  97. case err != nil:
  98. return
  99. case overflow:
  100. err = fmt.Errorf("line %d too long", rdr.Lineno)
  101. return
  102. }
  103. rdr.Lineno++
  104. line = bytes.TrimSpace(line)
  105. if len(line) == 0 {
  106. tok.Type = tokenComment
  107. return
  108. }
  109. switch line[0] {
  110. case '#':
  111. tok.Type = tokenComment
  112. return
  113. case '[':
  114. closing := bytes.IndexByte(line, ']')
  115. if closing < 0 {
  116. err = fmt.Errorf("missing ']' is section name at line %d", rdr.Lineno)
  117. return
  118. }
  119. tok.Name = string(line[1:closing])
  120. if closing+1 < len(line) {
  121. err = fmt.Errorf("trailing characters %q after section name %s",
  122. line[closing+1:], tok.Name)
  123. return
  124. }
  125. return
  126. default:
  127. tok.Type = tokenKey
  128. if idx := bytes.IndexByte(line, '='); idx >= 0 {
  129. optname := bytes.TrimSpace(line[:idx])
  130. values := bytes.Split(line[idx+1:], []byte{' '})
  131. tok.Name = string(optname)
  132. tok.Values = make([]string, 0, len(values))
  133. for _, word := range values {
  134. word = bytes.TrimSpace(word)
  135. if len(word) > 0 {
  136. tok.Values = append(tok.Values, string(word))
  137. }
  138. }
  139. } else {
  140. // boolean option
  141. tok.Name = string(line)
  142. tok.Values = nil
  143. }
  144. return
  145. }
  146. }
  147. func ParseConfig(r io.Reader) (conf PacmanConfig, err error) {
  148. rdr := newConfReader(r)
  149. rdrStack := []confReader{rdr}
  150. conf.SetDefaults()
  151. confReflect := reflect.ValueOf(&conf).Elem()
  152. var currentSection string
  153. var curRepo *RepoConfig
  154. lineloop:
  155. for {
  156. line, err := rdr.ParseLine()
  157. // fmt.Printf("%+v\n", line)
  158. switch err {
  159. case io.EOF:
  160. // pop reader stack.
  161. l := len(rdrStack)
  162. if l == 1 {
  163. break lineloop
  164. }
  165. rdr = rdrStack[l-2]
  166. rdrStack = rdrStack[:l-1]
  167. default:
  168. break lineloop
  169. case nil:
  170. // Ok.
  171. }
  172. switch line.Type {
  173. case tokenComment:
  174. case tokenSection:
  175. currentSection = line.Name
  176. if currentSection != "options" {
  177. conf.Repos = append(conf.Repos, RepoConfig{})
  178. curRepo = &conf.Repos[len(conf.Repos)-1]
  179. curRepo.Name = line.Name
  180. }
  181. case tokenKey:
  182. switch line.Name {
  183. case "SigLevel":
  184. // TODO: implement SigLevel parsing.
  185. continue lineloop
  186. case "Usage":
  187. for _, usage := range line.Values {
  188. switch usage {
  189. case "Sync":
  190. curRepo.Usage |= UsageSync
  191. case "Search":
  192. curRepo.Usage |= UsageSearch
  193. case "Install":
  194. curRepo.Usage |= UsageInstall
  195. case "Upgrade":
  196. curRepo.Usage |= UsageUpgrade
  197. case "All":
  198. curRepo.Usage |= UsageAll
  199. default:
  200. err = fmt.Errorf("unknown option at line %d: %s", rdr.Lineno, line.Name)
  201. break lineloop
  202. }
  203. }
  204. case "Server":
  205. curRepo.Servers = append(curRepo.Servers, line.Values...)
  206. continue lineloop
  207. case "Include":
  208. conf.Include = append(conf.Include, line.Values[0])
  209. f, err := os.Open(line.Values[0])
  210. if err != nil {
  211. err = fmt.Errorf("error while processing Include directive at line %d: %s",
  212. rdr.Lineno, err)
  213. break lineloop
  214. }
  215. rdr = newConfReader(f)
  216. rdrStack = append(rdrStack, rdr)
  217. continue lineloop
  218. case "UseDelta":
  219. if len(line.Values) > 0 {
  220. deltaRatio, err := strconv.ParseFloat(line.Values[0], 64)
  221. if err != nil {
  222. break lineloop
  223. }
  224. conf.UseDelta = deltaRatio
  225. }
  226. continue lineloop
  227. }
  228. if currentSection != "options" {
  229. err = fmt.Errorf("option %s outside of [options] section, at line %d",
  230. line.Name, rdr.Lineno)
  231. break lineloop
  232. }
  233. // main options.
  234. if opt, ok := optionsMap[line.Name]; ok {
  235. // boolean option.
  236. conf.Options |= opt
  237. } else {
  238. // key-value option.
  239. fld := confReflect.FieldByName(line.Name)
  240. if !fld.IsValid() || !fld.CanAddr() {
  241. _ = fmt.Errorf("unknown option at line %d: %s", rdr.Lineno, line.Name)
  242. continue
  243. }
  244. switch fieldP := fld.Addr().Interface().(type) {
  245. case *string:
  246. // single valued option.
  247. *fieldP = strings.Join(line.Values, " ")
  248. case *[]string:
  249. //many valued option.
  250. *fieldP = append(*fieldP, line.Values...)
  251. }
  252. }
  253. }
  254. }
  255. if len(conf.CleanMethod) == 0 {
  256. conf.CleanMethod = []string{"KeepInstalled"}
  257. }
  258. if len(conf.CacheDir) == 0 {
  259. conf.CacheDir = []string{"/var/cache/pacman/pkg/"} //should only be set if the config does not specify this
  260. }
  261. for n, _ := range conf.Repos {
  262. repo := &conf.Repos[n]
  263. if repo.Usage == 0 {
  264. repo.Usage = UsageAll
  265. }
  266. }
  267. return conf, err
  268. }
  269. func (conf *PacmanConfig) SetDefaults() {
  270. conf.RootDir = "/"
  271. conf.DBPath = "/var/lib/pacman"
  272. conf.DBPath = "/var/lib/pacman/"
  273. conf.HookDir = []string{"/etc/pacman.d/hooks/"} //should be added to whatever the config states
  274. conf.GPGDir = "/etc/pacman.d/gnupg/"
  275. conf.LogFile = "/var/log/pacman.log"
  276. conf.UseDelta = 0.7
  277. conf.SigLevel = SigPackage | SigPackageOptional | SigDatabase | SigDatabaseOptional
  278. conf.LocalFileSigLevel = SigUseDefault
  279. conf.RemoteFileSigLevel = SigUseDefault
  280. }
  281. func getArch() (string, error) {
  282. var uname syscall.Utsname
  283. err := syscall.Uname(&uname)
  284. if err != nil {
  285. return "", err
  286. }
  287. var arch [65]byte
  288. for i, c := range uname.Machine {
  289. if c == 0 {
  290. return string(arch[:i]), nil
  291. }
  292. arch[i] = byte(c)
  293. }
  294. return string(arch[:]), nil
  295. }
  296. func (conf *PacmanConfig) CreateHandle() (*Handle, error) {
  297. h, err := Init(conf.RootDir, conf.DBPath)
  298. if err != nil {
  299. return nil, err
  300. }
  301. if conf.Architecture == "auto" {
  302. conf.Architecture, err = getArch()
  303. if err != nil {
  304. return nil, fmt.Errorf("architecture is 'auto' but couldn't uname()")
  305. }
  306. }
  307. for _, repoconf := range conf.Repos {
  308. // TODO: set SigLevel
  309. db, err := h.RegisterSyncDb(repoconf.Name, 0)
  310. if err == nil {
  311. for i, addr := range repoconf.Servers {
  312. addr = strings.Replace(addr, "$repo", repoconf.Name, -1)
  313. addr = strings.Replace(addr, "$arch", conf.Architecture, -1)
  314. repoconf.Servers[i] = addr
  315. }
  316. db.SetServers(repoconf.Servers)
  317. db.SetUsage(repoconf.Usage)
  318. }
  319. }
  320. err = h.SetCacheDirs(conf.CacheDir...)
  321. if err != nil {
  322. return nil, err
  323. }
  324. // add hook directories 1-by-1 to avoid overwriting the system directory
  325. for _, dir := range conf.HookDir {
  326. err = h.AddHookDir(dir)
  327. if err != nil {
  328. return nil, err
  329. }
  330. }
  331. err = h.SetGPGDir(conf.GPGDir)
  332. if err != nil {
  333. return nil, err
  334. }
  335. err = h.SetLogFile(conf.LogFile)
  336. if err != nil {
  337. return nil, err
  338. }
  339. err = h.SetIgnorePkgs(conf.IgnorePkg...)
  340. if err != nil {
  341. return nil, err
  342. }
  343. err = h.SetIgnoreGroups(conf.IgnoreGroup...)
  344. if err != nil {
  345. return nil, err
  346. }
  347. err = h.SetArch(conf.Architecture)
  348. if err != nil {
  349. return nil, err
  350. }
  351. h.SetNoUpgrades(conf.NoUpgrade...)
  352. if err != nil {
  353. return nil, err
  354. }
  355. h.SetNoExtracts(conf.NoExtract...)
  356. if err != nil {
  357. return nil, err
  358. }
  359. err = h.SetDefaultSigLevel(conf.SigLevel)
  360. if err != nil {
  361. return nil, err
  362. }
  363. err = h.SetLocalFileSigLevel(conf.LocalFileSigLevel)
  364. if err != nil {
  365. return nil, err
  366. }
  367. err = h.SetRemoteFileSigLevel(conf.RemoteFileSigLevel)
  368. if err != nil {
  369. return nil, err
  370. }
  371. err = h.SetDeltaRatio(conf.UseDelta)
  372. if err != nil {
  373. return nil, err
  374. }
  375. err = h.SetUseSyslog(conf.Options&ConfUseSyslog > 0)
  376. if err != nil {
  377. return nil, err
  378. }
  379. err = h.SetCheckSpace(conf.Options&ConfCheckSpace > 0)
  380. if err != nil {
  381. return nil, err
  382. }
  383. return h, nil
  384. }