handlers.go 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452
  1. package samba
  2. import (
  3. "encoding/json"
  4. "log"
  5. "net/http"
  6. "path/filepath"
  7. "strings"
  8. "imuslab.com/arozos/mod/utils"
  9. )
  10. // Get current samba service status
  11. func (s *ShareManager) SmbdStates(w http.ResponseWriter, r *http.Request) {
  12. set, err := utils.PostPara(r, "set")
  13. if err != nil {
  14. //return the current smbd states
  15. smbdRunning, err := IsSmbdRunning()
  16. if err != nil {
  17. utils.SendErrorResponse(w, err.Error())
  18. return
  19. }
  20. js, _ := json.Marshal(smbdRunning)
  21. utils.SendJSONResponse(w, string(js))
  22. return
  23. } else if set == "enable" {
  24. //Set smbd to enable
  25. err = SetSmbdEnableState(true)
  26. if err != nil {
  27. utils.SendErrorResponse(w, err.Error())
  28. return
  29. }
  30. utils.SendOK(w)
  31. return
  32. } else if set == "disable" {
  33. //Set smbd to disable
  34. err = SetSmbdEnableState(false)
  35. if err != nil {
  36. utils.SendErrorResponse(w, err.Error())
  37. return
  38. }
  39. utils.SendOK(w)
  40. return
  41. }
  42. utils.SendErrorResponse(w, "not support set state: "+set+". Only support enable /disable")
  43. }
  44. // List all the samba shares
  45. func (s *ShareManager) ListSambaShares(w http.ResponseWriter, r *http.Request) {
  46. shares, err := s.ReadSambaShares()
  47. if err != nil {
  48. utils.SendErrorResponse(w, err.Error())
  49. return
  50. }
  51. //Remove those shares that is reserved by systems
  52. shares = s.FilterSystemCreatedShares(shares)
  53. js, _ := json.Marshal(shares)
  54. utils.SendJSONResponse(w, string(js))
  55. }
  56. // Add a samba share
  57. func (s *ShareManager) AddSambaShare(w http.ResponseWriter, r *http.Request) {
  58. shareName, err := utils.PostPara(r, "name")
  59. if err != nil {
  60. utils.SendErrorResponse(w, "share name not given")
  61. return
  62. }
  63. shareName = strings.TrimSpace(shareName)
  64. //Check if this share name already been used
  65. shareNameExists, err := s.ShareNameExists(shareName)
  66. if err != nil {
  67. utils.SendErrorResponse(w, err.Error())
  68. return
  69. }
  70. if shareNameExists {
  71. utils.SendErrorResponse(w, "share with identical name already exists")
  72. return
  73. }
  74. sharePath, err := utils.PostPara(r, "path")
  75. if err != nil {
  76. utils.SendErrorResponse(w, "share path not given")
  77. return
  78. }
  79. //Parse the path to absolute path
  80. absoluteSharePath, err := filepath.Abs(sharePath)
  81. if err != nil {
  82. utils.SendErrorResponse(w, err.Error())
  83. return
  84. }
  85. //Check if path exists
  86. if !utils.FileExists(absoluteSharePath) {
  87. utils.SendErrorResponse(w, "target path not exists")
  88. return
  89. }
  90. //Check if target path is a folder
  91. if !utils.IsDir(absoluteSharePath) {
  92. utils.SendErrorResponse(w, "target path is not a directory")
  93. return
  94. }
  95. //Check if it is a reserved / protected path
  96. if isPathInsideImportantFolders(absoluteSharePath) {
  97. utils.SendErrorResponse(w, "system reserved path cannot be shared")
  98. return
  99. }
  100. validUsersJSON, err := utils.PostPara(r, "users")
  101. if err != nil {
  102. utils.SendErrorResponse(w, "no valid user givens")
  103. return
  104. }
  105. //Parse valid users into string slice
  106. validUsers := []string{}
  107. err = json.Unmarshal([]byte(validUsersJSON), &validUsers)
  108. if err != nil {
  109. utils.SendErrorResponse(w, "unable to parse JSON for valid users")
  110. return
  111. }
  112. //Check if all the users exists in the host OS
  113. for _, validUser := range validUsers {
  114. thisUnixUserExists, err := s.SambaUserExists(validUser)
  115. if err != nil {
  116. utils.SendErrorResponse(w, err.Error())
  117. return
  118. }
  119. if !thisUnixUserExists {
  120. //This user not exists
  121. utils.SendErrorResponse(w, validUser+" is not a valid unix user")
  122. return
  123. }
  124. }
  125. readOnly, err := utils.PostBool(r, "readonly")
  126. if err != nil {
  127. readOnly = false
  128. }
  129. browseable, err := utils.PostBool(r, "browseable")
  130. if err != nil {
  131. browseable = true
  132. }
  133. allowGuest, err := utils.PostBool(r, "guestok")
  134. if err != nil {
  135. allowGuest = false
  136. }
  137. shareToCreate := ShareConfig{
  138. Name: shareName,
  139. Path: absoluteSharePath,
  140. ValidUsers: validUsers,
  141. ReadOnly: readOnly,
  142. Browseable: browseable,
  143. GuestOk: allowGuest,
  144. }
  145. //Add the new share to smb.conf
  146. err = s.CreateNewSambaShare(&shareToCreate)
  147. if err != nil {
  148. utils.SendErrorResponse(w, err.Error())
  149. return
  150. }
  151. //Restart smbd
  152. err = restartSmbd()
  153. if err != nil {
  154. utils.SendErrorResponse(w, err.Error())
  155. return
  156. }
  157. utils.SendOK(w)
  158. }
  159. // Remove a samba share by name
  160. func (s *ShareManager) DelSambaShare(w http.ResponseWriter, r *http.Request) {
  161. shareName, err := utils.PostPara(r, "name")
  162. if err != nil {
  163. utils.SendErrorResponse(w, "share name not given")
  164. return
  165. }
  166. //Check if share exists
  167. shareExists, err := s.ShareExists(shareName)
  168. if err != nil {
  169. utils.SendErrorResponse(w, err.Error())
  170. return
  171. }
  172. if !shareExists {
  173. utils.SendErrorResponse(w, "share to be remove not exists")
  174. return
  175. }
  176. //Remove the share from config file
  177. err = s.RemoveSambaShareConfig(shareName)
  178. if err != nil {
  179. utils.SendErrorResponse(w, err.Error())
  180. return
  181. }
  182. //Restart smbd
  183. err = restartSmbd()
  184. if err != nil {
  185. utils.SendErrorResponse(w, err.Error())
  186. return
  187. }
  188. utils.SendOK(w)
  189. }
  190. // Add a new samba user
  191. func (s *ShareManager) NewSambaUser(w http.ResponseWriter, r *http.Request) {
  192. username, err := utils.PostPara(r, "username")
  193. if err != nil {
  194. utils.SendErrorResponse(w, "username not given")
  195. return
  196. }
  197. password, err := utils.PostPara(r, "password")
  198. if err != nil {
  199. utils.SendErrorResponse(w, "password not set")
  200. return
  201. }
  202. err = s.AddSambaUser(username, password)
  203. if err != nil {
  204. utils.SendErrorResponse(w, err.Error())
  205. return
  206. }
  207. utils.SendOK(w)
  208. }
  209. // Remove a samba user, check for admin before calling
  210. func (s *ShareManager) DelSambaUser(w http.ResponseWriter, r *http.Request) {
  211. username, err := utils.PostPara(r, "username")
  212. if err != nil {
  213. utils.SendErrorResponse(w, "username not given")
  214. return
  215. }
  216. //Remove the samba user
  217. err = s.RemoveSmbUser(username)
  218. if err != nil {
  219. utils.SendErrorResponse(w, err.Error())
  220. return
  221. }
  222. utils.SendOK(w)
  223. }
  224. // List all samba users info
  225. func (s *ShareManager) ListSambaUsers(w http.ResponseWriter, r *http.Request) {
  226. type SimplifiedUserInfo struct {
  227. UnixUsername string
  228. Domain string
  229. IsArozOSUser bool
  230. }
  231. results := []*SimplifiedUserInfo{}
  232. userInfo, err := s.ListSambaUsersInfo()
  233. if err != nil {
  234. utils.SendErrorResponse(w, err.Error())
  235. return
  236. }
  237. for _, thisUserInfo := range userInfo {
  238. thisIsArozOSUser := s.UserHandler.GetAuthAgent().UserExists(strings.TrimSpace(thisUserInfo.UnixUsername))
  239. results = append(results, &SimplifiedUserInfo{
  240. UnixUsername: strings.TrimSpace(thisUserInfo.UnixUsername),
  241. Domain: thisUserInfo.Domain,
  242. IsArozOSUser: thisIsArozOSUser,
  243. })
  244. }
  245. js, _ := json.Marshal(results)
  246. utils.SendJSONResponse(w, string(js))
  247. }
  248. // Activate a user account from arozos into samba user
  249. func (s *ShareManager) ActivateUserAccount(w http.ResponseWriter, r *http.Request, password string) {
  250. userInfo, _ := s.UserHandler.GetUserInfoFromRequest(w, r)
  251. //Register this user to samba if not exists
  252. sambaUserExists, err := s.SambaUserExists(userInfo.Username)
  253. if err != nil {
  254. utils.SendErrorResponse(w, err.Error())
  255. return
  256. }
  257. if !sambaUserExists {
  258. //This user account not activated yet. Activate it
  259. err = s.AddSambaUser(userInfo.Username, password)
  260. if err != nil {
  261. utils.SendErrorResponse(w, err.Error())
  262. return
  263. }
  264. }
  265. //Create the user root share folders
  266. for _, fsh := range userInfo.GetAllAccessibleFileSystemHandler() {
  267. if fsh.IsNetworkDrive() {
  268. //Samba can only work with drives locally hosted on this server
  269. //Skip network drives
  270. continue
  271. }
  272. fshID := fsh.UUID
  273. fshSharePath := fsh.Path
  274. if fsh.RequierUserIsolation() {
  275. //User seperated storage. Only mount the user one
  276. fshID = userInfo.Username + "_" + fsh.UUID
  277. fshSharePath = filepath.Join(fsh.Path, "/users/", userInfo.Username+"/")
  278. }
  279. fshID = sanitizeShareName(fshID)
  280. //Check if the share already exists
  281. shareExists, err := s.ShareExists(fshID)
  282. if err != nil {
  283. continue
  284. }
  285. if !shareExists {
  286. //Try to create the share
  287. fshShareAbsolutePath, err := filepath.Abs(fshSharePath)
  288. if err != nil {
  289. log.Println("[Samba] Unable to generate share config for path: " + fshSharePath)
  290. continue
  291. }
  292. //Check if that folder exists
  293. if !utils.FileExists(fshShareAbsolutePath) {
  294. //Folder not exists. Continue
  295. log.Println("[Samba] Path not exists for file system handler: " + fshSharePath)
  296. continue
  297. }
  298. //Ok! Create the share with this username
  299. err = s.CreateNewSambaShare(&ShareConfig{
  300. Name: fshID,
  301. Path: fshShareAbsolutePath,
  302. ValidUsers: []string{userInfo.Username},
  303. ReadOnly: false,
  304. Browseable: !fsh.RequierUserIsolation(),
  305. GuestOk: false,
  306. })
  307. if err != nil {
  308. log.Println("[Samba] Failed to create share: " + err.Error())
  309. utils.SendErrorResponse(w, err.Error())
  310. return
  311. }
  312. } else {
  313. //Share exists. Add this user to such share
  314. err = s.AddUserToSambaShare(fshID, userInfo.Username)
  315. if err != nil {
  316. log.Println("[Samba] Failed to add user " + userInfo.Username + " to share " + fshID + ": " + err.Error())
  317. utils.SendErrorResponse(w, err.Error())
  318. return
  319. }
  320. }
  321. }
  322. utils.SendOK(w)
  323. }
  324. // Get if the user share has been enabled
  325. func (s *ShareManager) HandleUserSmbStatusList(w http.ResponseWriter, r *http.Request) {
  326. type UserStatus struct {
  327. SmbdEnabled bool
  328. UserSmbShareEnabled bool
  329. UserSmbShareList []*ShareConfig
  330. }
  331. result := UserStatus{
  332. SmbdEnabled: s.IsEnabled(),
  333. UserSmbShareEnabled: false,
  334. UserSmbShareList: []*ShareConfig{},
  335. }
  336. userInfo, err := s.UserHandler.GetUserInfoFromRequest(w, r)
  337. if err != nil {
  338. utils.SendErrorResponse(w, err.Error())
  339. return
  340. }
  341. userAccessibleShares, err := s.GetUsersShare(userInfo.Username)
  342. if err != nil {
  343. //User never used smb service
  344. js, _ := json.Marshal(result)
  345. utils.SendJSONResponse(w, string(js))
  346. return
  347. }
  348. if len(userAccessibleShares) == 0 {
  349. result.UserSmbShareEnabled = false
  350. } else {
  351. result.UserSmbShareEnabled = true
  352. result.UserSmbShareList = userAccessibleShares
  353. }
  354. js, _ := json.Marshal(result)
  355. utils.SendJSONResponse(w, string(js))
  356. }
  357. // Deactivate the user account by removing user access to all shares
  358. func (s *ShareManager) DeactiveUserAccount(w http.ResponseWriter, r *http.Request) {
  359. userInfo, err := s.UserHandler.GetUserInfoFromRequest(w, r)
  360. if err != nil {
  361. utils.SendErrorResponse(w, err.Error())
  362. return
  363. }
  364. userAccessibleShares, err := s.GetUsersShare(userInfo.Username)
  365. if err != nil {
  366. utils.SendErrorResponse(w, err.Error())
  367. return
  368. }
  369. //For each of the shares this user can access, remove his name from the share
  370. for _, userAccessibleShare := range userAccessibleShares {
  371. err = s.RemoveUserFromSambaShare(userAccessibleShare.Name, userInfo.Username)
  372. if err != nil {
  373. log.Println("[Samba] Unable to remove user " + userInfo.Username + " from share: " + err.Error())
  374. continue
  375. }
  376. }
  377. //Remove this samba user
  378. err = s.RemoveSmbUser(userInfo.Username)
  379. if err != nil {
  380. utils.SendErrorResponse(w, "Samba user remove failed: "+err.Error())
  381. return
  382. }
  383. utils.SendOK(w)
  384. }