handlers.go 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578
  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. err = restartSmbd()
  323. if err != nil {
  324. utils.SendErrorResponse(w, err.Error())
  325. return
  326. }
  327. utils.SendOK(w)
  328. }
  329. // Get if the user share has been enabled
  330. func (s *ShareManager) HandleUserSmbStatusList(w http.ResponseWriter, r *http.Request) {
  331. type UserStatus struct {
  332. SmbdEnabled bool
  333. UserSmbShareEnabled bool
  334. UserSmbShareList []*ShareConfig
  335. }
  336. result := UserStatus{
  337. SmbdEnabled: s.IsEnabled(),
  338. UserSmbShareEnabled: false,
  339. UserSmbShareList: []*ShareConfig{},
  340. }
  341. userInfo, err := s.UserHandler.GetUserInfoFromRequest(w, r)
  342. if err != nil {
  343. utils.SendErrorResponse(w, err.Error())
  344. return
  345. }
  346. userAccessibleShares, err := s.GetUsersShare(userInfo.Username)
  347. if err != nil {
  348. //User never used smb service
  349. js, _ := json.Marshal(result)
  350. utils.SendJSONResponse(w, string(js))
  351. return
  352. }
  353. if len(userAccessibleShares) == 0 {
  354. result.UserSmbShareEnabled = false
  355. } else {
  356. result.UserSmbShareEnabled = true
  357. result.UserSmbShareList = userAccessibleShares
  358. }
  359. js, _ := json.Marshal(result)
  360. utils.SendJSONResponse(w, string(js))
  361. }
  362. // Deactivate the user account by removing user access to all shares
  363. func (s *ShareManager) DeactiveUserAccount(w http.ResponseWriter, r *http.Request) {
  364. userInfo, err := s.UserHandler.GetUserInfoFromRequest(w, r)
  365. if err != nil {
  366. utils.SendErrorResponse(w, err.Error())
  367. return
  368. }
  369. userAccessibleShares, err := s.GetUsersShare(userInfo.Username)
  370. if err != nil {
  371. utils.SendErrorResponse(w, err.Error())
  372. return
  373. }
  374. //For each of the shares this user can access, remove his name from the share
  375. for _, userAccessibleShare := range userAccessibleShares {
  376. err = s.RemoveUserFromSambaShare(userAccessibleShare.Name, userInfo.Username)
  377. if err != nil {
  378. log.Println("[Samba] Unable to remove user " + userInfo.Username + " from share: " + err.Error())
  379. continue
  380. }
  381. }
  382. //Remove this samba user
  383. err = s.RemoveSmbUser(userInfo.Username)
  384. if err != nil {
  385. utils.SendErrorResponse(w, "Samba user remove failed: "+err.Error())
  386. return
  387. }
  388. err = restartSmbd()
  389. if err != nil {
  390. utils.SendErrorResponse(w, err.Error())
  391. return
  392. }
  393. utils.SendOK(w)
  394. }
  395. // Handle update to accessible users
  396. func (s *ShareManager) HandleAccessUserUpdate(w http.ResponseWriter, r *http.Request) {
  397. shareName, err := utils.PostPara(r, "name")
  398. if err != nil {
  399. utils.SendErrorResponse(w, "share name not given")
  400. return
  401. }
  402. newUserListJSON, err := utils.PostPara(r, "users")
  403. if err != nil {
  404. utils.SendErrorResponse(w, "list of new users not given")
  405. return
  406. }
  407. //Parse the user list from json string to string slice
  408. newUserList := []string{}
  409. err = json.Unmarshal([]byte(newUserListJSON), &newUserList)
  410. if err != nil {
  411. log.Println("[Samba] Parse new user list failed: " + err.Error())
  412. utils.SendErrorResponse(w, "failed to parse the new user list")
  413. return
  414. }
  415. //read the target share from smb.conf
  416. targetShare, err := s.GetShareByName(shareName)
  417. if err != nil {
  418. utils.SendErrorResponse(w, err.Error())
  419. return
  420. }
  421. for _, originalUser := range targetShare.ValidUsers {
  422. if !utils.StringInArray(newUserList, originalUser) {
  423. //This user is not longer allowed to access this share
  424. //remove this user from this share
  425. s.RemoveUserFromSambaShare(shareName, originalUser)
  426. }
  427. }
  428. for _, newUsername := range newUserList {
  429. if !s.UserCanAccessShare(targetShare, newUsername) {
  430. err = s.AddUserToSambaShare(targetShare.Name, newUsername)
  431. if err != nil {
  432. utils.SendErrorResponse(w, err.Error())
  433. return
  434. }
  435. }
  436. }
  437. err = restartSmbd()
  438. if err != nil {
  439. utils.SendErrorResponse(w, err.Error())
  440. return
  441. }
  442. utils.SendOK(w)
  443. }
  444. // Handle changing path of share
  445. func (s *ShareManager) HandleSharePathChange(w http.ResponseWriter, r *http.Request) {
  446. shareName, err := utils.PostPara(r, "name")
  447. if err != nil {
  448. utils.SendErrorResponse(w, "share name not given")
  449. return
  450. }
  451. newSharePath, err := utils.PostPara(r, "path")
  452. if err != nil {
  453. utils.SendErrorResponse(w, "list of new users not given")
  454. return
  455. }
  456. //Convert path to absolute and check if folder exists
  457. newSharePathAbsolute, err := filepath.Abs(newSharePath)
  458. if err != nil {
  459. utils.SendErrorResponse(w, err.Error())
  460. return
  461. }
  462. if !utils.FileExists(newSharePathAbsolute) {
  463. utils.SendErrorResponse(w, "target folder not exists")
  464. return
  465. }
  466. //Check if path sharing is allowed
  467. if isPathInsideImportantFolders(newSharePathAbsolute) {
  468. utils.SendErrorResponse(w, "path is or inside protected folders")
  469. return
  470. }
  471. //read the target share from smb.conf
  472. targetShare, err := s.GetShareByName(shareName)
  473. if err != nil {
  474. utils.SendErrorResponse(w, err.Error())
  475. return
  476. }
  477. //Update and save share to smb.conf
  478. targetShare.Path = newSharePathAbsolute
  479. err = targetShare.SaveToConfig()
  480. if err != nil {
  481. utils.SendErrorResponse(w, err.Error())
  482. return
  483. }
  484. //Restart smbd
  485. err = restartSmbd()
  486. if err != nil {
  487. utils.SendErrorResponse(w, err.Error())
  488. return
  489. }
  490. utils.SendOK(w)
  491. }