cmd_folders.go 9.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361
  1. // Copyright (C) 2014 Audrius Butkevičius
  2. package main
  3. import (
  4. "fmt"
  5. "path/filepath"
  6. "strings"
  7. "github.com/AudriusButkevicius/cli"
  8. "github.com/syncthing/syncthing/lib/config"
  9. "github.com/syncthing/syncthing/lib/fs"
  10. )
  11. func init() {
  12. cliCommands = append(cliCommands, cli.Command{
  13. Name: "folders",
  14. HideHelp: true,
  15. Usage: "Folder command group",
  16. Subcommands: []cli.Command{
  17. {
  18. Name: "list",
  19. Usage: "List available folders",
  20. Requires: &cli.Requires{},
  21. Action: foldersList,
  22. },
  23. {
  24. Name: "add",
  25. Usage: "Add a new folder",
  26. Requires: &cli.Requires{"folder id", "directory"},
  27. Action: foldersAdd,
  28. },
  29. {
  30. Name: "remove",
  31. Usage: "Remove an existing folder",
  32. Requires: &cli.Requires{"folder id"},
  33. Action: foldersRemove,
  34. },
  35. {
  36. Name: "override",
  37. Usage: "Override changes from other nodes for a master folder",
  38. Requires: &cli.Requires{"folder id"},
  39. Action: foldersOverride,
  40. },
  41. {
  42. Name: "get",
  43. Usage: "Get a property of a folder",
  44. Requires: &cli.Requires{"folder id", "property"},
  45. Action: foldersGet,
  46. },
  47. {
  48. Name: "set",
  49. Usage: "Set a property of a folder",
  50. Requires: &cli.Requires{"folder id", "property", "value..."},
  51. Action: foldersSet,
  52. },
  53. {
  54. Name: "unset",
  55. Usage: "Unset a property of a folder",
  56. Requires: &cli.Requires{"folder id", "property"},
  57. Action: foldersUnset,
  58. },
  59. {
  60. Name: "devices",
  61. Usage: "Folder devices command group",
  62. HideHelp: true,
  63. Subcommands: []cli.Command{
  64. {
  65. Name: "list",
  66. Usage: "List of devices which the folder is shared with",
  67. Requires: &cli.Requires{"folder id"},
  68. Action: foldersDevicesList,
  69. },
  70. {
  71. Name: "add",
  72. Usage: "Share a folder with a device",
  73. Requires: &cli.Requires{"folder id", "device id"},
  74. Action: foldersDevicesAdd,
  75. },
  76. {
  77. Name: "remove",
  78. Usage: "Unshare a folder with a device",
  79. Requires: &cli.Requires{"folder id", "device id"},
  80. Action: foldersDevicesRemove,
  81. },
  82. {
  83. Name: "clear",
  84. Usage: "Unshare a folder with all devices",
  85. Requires: &cli.Requires{"folder id"},
  86. Action: foldersDevicesClear,
  87. },
  88. },
  89. },
  90. },
  91. })
  92. }
  93. func foldersList(c *cli.Context) {
  94. cfg := getConfig(c)
  95. first := true
  96. writer := newTableWriter()
  97. for _, folder := range cfg.Folders {
  98. if !first {
  99. fmt.Fprintln(writer)
  100. }
  101. fs := folder.Filesystem()
  102. fmt.Fprintln(writer, "ID:\t", folder.ID, "\t")
  103. fmt.Fprintln(writer, "Path:\t", fs.URI(), "\t(directory)")
  104. fmt.Fprintln(writer, "Path type:\t", fs.Type(), "\t(directory-type)")
  105. fmt.Fprintln(writer, "Folder type:\t", folder.Type, "\t(type)")
  106. fmt.Fprintln(writer, "Ignore permissions:\t", folder.IgnorePerms, "\t(permissions)")
  107. fmt.Fprintln(writer, "Rescan interval in seconds:\t", folder.RescanIntervalS, "\t(rescan)")
  108. if folder.Versioning.Type != "" {
  109. fmt.Fprintln(writer, "Versioning:\t", folder.Versioning.Type, "\t(versioning)")
  110. for key, value := range folder.Versioning.Params {
  111. fmt.Fprintf(writer, "Versioning %s:\t %s \t(versioning-%s)\n", key, value, key)
  112. }
  113. }
  114. first = false
  115. }
  116. writer.Flush()
  117. }
  118. func foldersAdd(c *cli.Context) {
  119. cfg := getConfig(c)
  120. abs, err := filepath.Abs(c.Args()[1])
  121. die(err)
  122. folder := config.FolderConfiguration{
  123. ID: c.Args()[0],
  124. Path: filepath.Clean(abs),
  125. FilesystemType: fs.FilesystemTypeBasic,
  126. }
  127. cfg.Folders = append(cfg.Folders, folder)
  128. setConfig(c, cfg)
  129. }
  130. func foldersRemove(c *cli.Context) {
  131. cfg := getConfig(c)
  132. rid := c.Args()[0]
  133. for i, folder := range cfg.Folders {
  134. if folder.ID == rid {
  135. last := len(cfg.Folders) - 1
  136. cfg.Folders[i] = cfg.Folders[last]
  137. cfg.Folders = cfg.Folders[:last]
  138. setConfig(c, cfg)
  139. return
  140. }
  141. }
  142. die("Folder " + rid + " not found")
  143. }
  144. func foldersOverride(c *cli.Context) {
  145. cfg := getConfig(c)
  146. rid := c.Args()[0]
  147. for _, folder := range cfg.Folders {
  148. if folder.ID == rid && folder.Type == config.FolderTypeSendOnly {
  149. response := httpPost(c, "db/override", "")
  150. if response.StatusCode != 200 {
  151. err := fmt.Sprint("Failed to override changes\nStatus code: ", response.StatusCode)
  152. body := string(responseToBArray(response))
  153. if body != "" {
  154. err += "\nBody: " + body
  155. }
  156. die(err)
  157. }
  158. return
  159. }
  160. }
  161. die("Folder " + rid + " not found or folder not master")
  162. }
  163. func foldersGet(c *cli.Context) {
  164. cfg := getConfig(c)
  165. rid := c.Args()[0]
  166. arg := strings.ToLower(c.Args()[1])
  167. for _, folder := range cfg.Folders {
  168. if folder.ID != rid {
  169. continue
  170. }
  171. if strings.HasPrefix(arg, "versioning-") {
  172. arg = arg[11:]
  173. value, ok := folder.Versioning.Params[arg]
  174. if ok {
  175. fmt.Println(value)
  176. return
  177. }
  178. die("Versioning property " + c.Args()[1][11:] + " not found")
  179. }
  180. switch arg {
  181. case "directory":
  182. fmt.Println(folder.Filesystem().URI())
  183. case "directory-type":
  184. fmt.Println(folder.Filesystem().Type())
  185. case "type":
  186. fmt.Println(folder.Type)
  187. case "permissions":
  188. fmt.Println(folder.IgnorePerms)
  189. case "rescan":
  190. fmt.Println(folder.RescanIntervalS)
  191. case "versioning":
  192. if folder.Versioning.Type != "" {
  193. fmt.Println(folder.Versioning.Type)
  194. }
  195. default:
  196. die("Invalid property: " + c.Args()[1] + "\nAvailable properties: directory, directory-type, type, permissions, versioning, versioning-<key>")
  197. }
  198. return
  199. }
  200. die("Folder " + rid + " not found")
  201. }
  202. func foldersSet(c *cli.Context) {
  203. rid := c.Args()[0]
  204. arg := strings.ToLower(c.Args()[1])
  205. val := strings.Join(c.Args()[2:], " ")
  206. cfg := getConfig(c)
  207. for i, folder := range cfg.Folders {
  208. if folder.ID != rid {
  209. continue
  210. }
  211. if strings.HasPrefix(arg, "versioning-") {
  212. cfg.Folders[i].Versioning.Params[arg[11:]] = val
  213. setConfig(c, cfg)
  214. return
  215. }
  216. switch arg {
  217. case "directory":
  218. cfg.Folders[i].Path = val
  219. case "directory-type":
  220. var fsType fs.FilesystemType
  221. fsType.UnmarshalText([]byte(val))
  222. cfg.Folders[i].FilesystemType = fsType
  223. case "type":
  224. var t config.FolderType
  225. if err := t.UnmarshalText([]byte(val)); err != nil {
  226. die("Invalid folder type: " + err.Error())
  227. }
  228. cfg.Folders[i].Type = t
  229. case "permissions":
  230. cfg.Folders[i].IgnorePerms = parseBool(val)
  231. case "rescan":
  232. cfg.Folders[i].RescanIntervalS = parseInt(val)
  233. case "versioning":
  234. cfg.Folders[i].Versioning.Type = val
  235. default:
  236. die("Invalid property: " + c.Args()[1] + "\nAvailable properties: directory, master, permissions, versioning, versioning-<key>")
  237. }
  238. setConfig(c, cfg)
  239. return
  240. }
  241. die("Folder " + rid + " not found")
  242. }
  243. func foldersUnset(c *cli.Context) {
  244. rid := c.Args()[0]
  245. arg := strings.ToLower(c.Args()[1])
  246. cfg := getConfig(c)
  247. for i, folder := range cfg.Folders {
  248. if folder.ID != rid {
  249. continue
  250. }
  251. if strings.HasPrefix(arg, "versioning-") {
  252. arg = arg[11:]
  253. if _, ok := folder.Versioning.Params[arg]; ok {
  254. delete(cfg.Folders[i].Versioning.Params, arg)
  255. setConfig(c, cfg)
  256. return
  257. }
  258. die("Versioning property " + c.Args()[1][11:] + " not found")
  259. }
  260. switch arg {
  261. case "versioning":
  262. cfg.Folders[i].Versioning.Type = ""
  263. cfg.Folders[i].Versioning.Params = make(map[string]string)
  264. default:
  265. die("Invalid property: " + c.Args()[1] + "\nAvailable properties: versioning, versioning-<key>")
  266. }
  267. setConfig(c, cfg)
  268. return
  269. }
  270. die("Folder " + rid + " not found")
  271. }
  272. func foldersDevicesList(c *cli.Context) {
  273. rid := c.Args()[0]
  274. cfg := getConfig(c)
  275. for _, folder := range cfg.Folders {
  276. if folder.ID != rid {
  277. continue
  278. }
  279. for _, device := range folder.Devices {
  280. fmt.Println(device.DeviceID)
  281. }
  282. return
  283. }
  284. die("Folder " + rid + " not found")
  285. }
  286. func foldersDevicesAdd(c *cli.Context) {
  287. rid := c.Args()[0]
  288. nid := parseDeviceID(c.Args()[1])
  289. cfg := getConfig(c)
  290. for i, folder := range cfg.Folders {
  291. if folder.ID != rid {
  292. continue
  293. }
  294. for _, device := range folder.Devices {
  295. if device.DeviceID == nid {
  296. die("Device " + c.Args()[1] + " is already part of this folder")
  297. }
  298. }
  299. for _, device := range cfg.Devices {
  300. if device.DeviceID == nid {
  301. cfg.Folders[i].Devices = append(folder.Devices, config.FolderDeviceConfiguration{
  302. DeviceID: device.DeviceID,
  303. })
  304. setConfig(c, cfg)
  305. return
  306. }
  307. }
  308. die("Device " + c.Args()[1] + " not found in device list")
  309. }
  310. die("Folder " + rid + " not found")
  311. }
  312. func foldersDevicesRemove(c *cli.Context) {
  313. rid := c.Args()[0]
  314. nid := parseDeviceID(c.Args()[1])
  315. cfg := getConfig(c)
  316. for ri, folder := range cfg.Folders {
  317. if folder.ID != rid {
  318. continue
  319. }
  320. for ni, device := range folder.Devices {
  321. if device.DeviceID == nid {
  322. last := len(folder.Devices) - 1
  323. cfg.Folders[ri].Devices[ni] = folder.Devices[last]
  324. cfg.Folders[ri].Devices = cfg.Folders[ri].Devices[:last]
  325. setConfig(c, cfg)
  326. return
  327. }
  328. }
  329. die("Device " + c.Args()[1] + " not found")
  330. }
  331. die("Folder " + rid + " not found")
  332. }
  333. func foldersDevicesClear(c *cli.Context) {
  334. rid := c.Args()[0]
  335. cfg := getConfig(c)
  336. for i, folder := range cfg.Folders {
  337. if folder.ID != rid {
  338. continue
  339. }
  340. cfg.Folders[i].Devices = []config.FolderDeviceConfiguration{}
  341. setConfig(c, cfg)
  342. return
  343. }
  344. die("Folder " + rid + " not found")
  345. }