samba.go 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417
  1. package samba
  2. import (
  3. "bufio"
  4. "errors"
  5. "fmt"
  6. "os"
  7. "runtime"
  8. "strings"
  9. "imuslab.com/arozos/mod/user"
  10. "imuslab.com/arozos/mod/utils"
  11. )
  12. /*
  13. Samba Share Warpper
  14. Note that this module only provide exposing of local disk / storage.
  15. This module do not handle providing the virtualized interface for samba
  16. */
  17. type ShareManager struct {
  18. SambaConfigPath string //Config file for samba, aka smb.conf
  19. UserHandler *user.UserHandler
  20. }
  21. type ShareConfig struct {
  22. Name string
  23. Path string
  24. ValidUsers []string
  25. ReadOnly bool
  26. Browseable bool
  27. GuestOk bool
  28. parent *ShareManager
  29. }
  30. func NewSambaShareManager(userHandler *user.UserHandler) (*ShareManager, error) {
  31. if runtime.GOOS == "linux" {
  32. //Check if samba installed
  33. if !utils.FileExists("/bin/smbcontrol") {
  34. return nil, errors.New("samba not installed")
  35. }
  36. } else {
  37. return nil, errors.New("platform not supported")
  38. }
  39. return &ShareManager{
  40. SambaConfigPath: "/etc/samba/smb.conf",
  41. UserHandler: userHandler,
  42. }, nil
  43. }
  44. // ReadSambaShares reads / lists the smb.conf file and extracts all existing shares
  45. func (s *ShareManager) ReadSambaShares() ([]ShareConfig, error) {
  46. file, err := os.Open(s.SambaConfigPath)
  47. if err != nil {
  48. return nil, err
  49. }
  50. defer file.Close()
  51. var shares []ShareConfig
  52. var currentShare *ShareConfig
  53. scanner := bufio.NewScanner(file)
  54. for scanner.Scan() {
  55. line := strings.TrimSpace(scanner.Text())
  56. // Check for section headers
  57. if strings.HasPrefix(line, "[") && strings.HasSuffix(line, "]") {
  58. if currentShare != nil {
  59. currentShare.parent = s
  60. shares = append(shares, *currentShare)
  61. }
  62. currentShare = &ShareConfig{
  63. Name: strings.Trim(line, "[]"),
  64. }
  65. continue
  66. }
  67. // Check if we are currently processing a share section
  68. if currentShare != nil {
  69. tokens := strings.SplitN(line, "=", 2)
  70. if len(tokens) != 2 {
  71. continue
  72. }
  73. key := strings.TrimSpace(tokens[0])
  74. value := strings.TrimSpace(tokens[1])
  75. switch key {
  76. case "path":
  77. currentShare.Path = value
  78. case "valid users":
  79. currentShare.ValidUsers = strings.Fields(value)
  80. case "read only":
  81. currentShare.ReadOnly = (value == "yes")
  82. case "browseable":
  83. currentShare.Browseable = (value == "yes")
  84. case "guest ok":
  85. currentShare.GuestOk = (value == "yes")
  86. }
  87. }
  88. }
  89. // Add the last share if there is one
  90. if currentShare != nil {
  91. currentShare.parent = s
  92. shares = append(shares, *currentShare)
  93. }
  94. // Check for scanner errors
  95. if err := scanner.Err(); err != nil {
  96. return nil, err
  97. }
  98. return shares, nil
  99. }
  100. // Check if a share name is already used
  101. func (s *ShareManager) ShareNameExists(sharename string) (bool, error) {
  102. allShares, err := s.ReadSambaShares()
  103. if err != nil {
  104. return false, err
  105. }
  106. for _, share := range allShares {
  107. if strings.EqualFold(share.Name, sharename) {
  108. return true, nil
  109. }
  110. }
  111. return false, nil
  112. }
  113. // A basic filter to remove system created smb shares entry in the list
  114. func (s *ShareManager) FilterSystemCreatedShares(shares []ShareConfig) []ShareConfig {
  115. namesToRemove := []string{"global", "printers", "print$"}
  116. filteredShares := []ShareConfig{}
  117. for _, share := range shares {
  118. if !utils.StringInArray(namesToRemove, share.Name) {
  119. filteredShares = append(filteredShares, share)
  120. }
  121. }
  122. return filteredShares
  123. }
  124. // CreateNewSambaShare converts the shareConfig to string and appends it to smb.conf if the share name does not already exist
  125. func (s *ShareManager) CreateNewSambaShare(shareToCreate *ShareConfig) error {
  126. // Path to smb.conf
  127. smbConfPath := s.SambaConfigPath
  128. // Open the smb.conf file for reading
  129. file, err := os.Open(smbConfPath)
  130. if err != nil {
  131. return fmt.Errorf("failed to open smb.conf: %v", err)
  132. }
  133. defer file.Close()
  134. // Check if the share already exists
  135. scanner := bufio.NewScanner(file)
  136. shareExists := false
  137. shareNameSection := fmt.Sprintf("[%s]", shareToCreate.Name)
  138. for scanner.Scan() {
  139. if strings.EqualFold(strings.TrimSpace(scanner.Text()), shareNameSection) {
  140. shareExists = true
  141. break
  142. }
  143. }
  144. if err := scanner.Err(); err != nil {
  145. return fmt.Errorf("failed to read smb.conf: %v", err)
  146. }
  147. if shareExists {
  148. return fmt.Errorf("share %s already exists", shareToCreate.Name)
  149. }
  150. // Convert ShareConfig to string
  151. shareConfigString := convertShareConfigToString(shareToCreate)
  152. // Open the smb.conf file for appending
  153. file, err = os.OpenFile(smbConfPath, os.O_APPEND|os.O_WRONLY, 0644)
  154. if err != nil {
  155. return fmt.Errorf("failed to open smb.conf for writing: %v", err)
  156. }
  157. defer file.Close()
  158. // Append the new share configuration
  159. if _, err := file.WriteString(shareConfigString); err != nil {
  160. return fmt.Errorf("failed to write to smb.conf: %v", err)
  161. }
  162. return nil
  163. }
  164. // RemoveSambaShareConfig removes the Samba share configuration from smb.conf
  165. func (s *ShareManager) RemoveSambaShareConfig(shareName string) error {
  166. // Check if the share exists
  167. shareExists, err := s.ShareExists(shareName)
  168. if err != nil {
  169. return err
  170. }
  171. if !shareExists {
  172. return errors.New("share not exists")
  173. }
  174. //Convert the sharename to correct case for matching
  175. shareName = s.getCorrectCaseForShareName(shareName)
  176. // Open the smb.conf file for reading
  177. file, err := os.Open(s.SambaConfigPath)
  178. if err != nil {
  179. return err
  180. }
  181. defer file.Close()
  182. // Create a temporary file to store modified smb.conf
  183. tmpFile, err := os.CreateTemp("", "smb.conf.*.tmp")
  184. if err != nil {
  185. return err
  186. }
  187. defer tmpFile.Close()
  188. // Create a scanner to read the smb.conf file line by line
  189. scanner := bufio.NewScanner(file)
  190. for scanner.Scan() {
  191. line := scanner.Text()
  192. // Check if the line contains the share name
  193. if strings.HasPrefix(line, "["+shareName+"]") {
  194. // Skip the lines until the next section
  195. for scanner.Scan() {
  196. checkingLine := scanner.Text()
  197. if strings.HasPrefix(checkingLine, "[") {
  198. //The header of the next section is also need to be kept
  199. fmt.Fprintln(tmpFile, checkingLine)
  200. break
  201. }
  202. }
  203. continue // Skip writing the share configuration to the temporary file
  204. }
  205. // Write the line to the temporary file
  206. _, err := fmt.Fprintln(tmpFile, line)
  207. if err != nil {
  208. return err
  209. }
  210. }
  211. // Check for scanner errors
  212. if err := scanner.Err(); err != nil {
  213. return err
  214. }
  215. // Close the original smb.conf file
  216. if err := file.Close(); err != nil {
  217. return err
  218. }
  219. // Close the temporary file
  220. if err := tmpFile.Close(); err != nil {
  221. return err
  222. }
  223. // Replace the original smb.conf file with the temporary file
  224. if err := os.Rename(tmpFile.Name(), "/etc/samba/smb.conf"); err != nil {
  225. return err
  226. }
  227. return nil
  228. }
  229. // ShareExists checks if a given share name exists in smb.conf
  230. func (s *ShareManager) ShareExists(shareName string) (bool, error) {
  231. // Path to smb.conf
  232. smbConfPath := s.SambaConfigPath
  233. // Open the smb.conf file for reading
  234. file, err := os.Open(smbConfPath)
  235. if err != nil {
  236. return false, fmt.Errorf("failed to open smb.conf: %v", err)
  237. }
  238. defer file.Close()
  239. // Check if the share already exists
  240. scanner := bufio.NewScanner(file)
  241. shareNameSection := fmt.Sprintf("[%s]", shareName)
  242. for scanner.Scan() {
  243. if strings.EqualFold(shareNameSection, strings.TrimSpace(scanner.Text())) {
  244. return true, nil
  245. }
  246. }
  247. if err := scanner.Err(); err != nil {
  248. return false, fmt.Errorf("failed to read smb.conf: %v", err)
  249. }
  250. return false, nil
  251. }
  252. // Add a new user to smb.conf share by name
  253. func (s *ShareManager) AddUserToSambaShare(shareName, username string) error {
  254. targetShare, err := s.GetShareByName(shareName)
  255. if err != nil {
  256. return err
  257. }
  258. if s.UserCanAccessShare(targetShare, username) {
  259. //User already have no access to this share, no need to delete
  260. return nil
  261. }
  262. //User not in this share. Append user name in this valid users
  263. targetShare.ValidUsers = append(targetShare.ValidUsers, username)
  264. //Delete the old one and create the new share with updated user list
  265. err = s.RemoveSambaShareConfig(shareName)
  266. if err != nil {
  267. return err
  268. }
  269. err = s.CreateNewSambaShare(targetShare)
  270. if err != nil {
  271. return err
  272. }
  273. return nil
  274. }
  275. // Return if the user can access this samba share
  276. func (s *ShareManager) UserCanAccessShare(targetSmbShare *ShareConfig, username string) bool {
  277. return utils.StringInArray(targetSmbShare.ValidUsers, username)
  278. }
  279. // Get a list of shares that this user have access to
  280. func (s *ShareManager) GetUsersShare(username string) ([]*ShareConfig, error) {
  281. allShares, err := s.ReadSambaShares()
  282. if err != nil {
  283. return nil, err
  284. }
  285. userAccessibleShares := []*ShareConfig{}
  286. for _, thisShare := range allShares {
  287. if s.UserCanAccessShare(&thisShare, username) {
  288. thisShareObject := thisShare
  289. userAccessibleShares = append(userAccessibleShares, &thisShareObject)
  290. }
  291. }
  292. return userAccessibleShares, nil
  293. }
  294. // Remove a user from smb.conf share by name
  295. func (s *ShareManager) RemoveUserFromSambaShare(shareName, username string) error {
  296. targetShare, err := s.GetShareByName(shareName)
  297. if err != nil {
  298. return err
  299. }
  300. if !s.UserCanAccessShare(targetShare, username) {
  301. //User already have no access to this share, no need to delete
  302. return nil
  303. }
  304. if len(targetShare.ValidUsers) == 1 && strings.EqualFold(targetShare.ValidUsers[0], username) {
  305. //This user is the only person who can access this share
  306. //Remove the share entirely
  307. //Delete the old one and create the new share with updated user list
  308. err = s.RemoveSambaShareConfig(shareName)
  309. if err != nil {
  310. return err
  311. }
  312. return nil
  313. }
  314. //User is in this share but this share contain other users.
  315. err = s.RemoveSambaShareConfig(shareName)
  316. if err != nil {
  317. return err
  318. }
  319. //Create a new valid user list
  320. newShareValidUsers := []string{}
  321. for _, validUser := range targetShare.ValidUsers {
  322. if !strings.EqualFold(validUser, username) {
  323. newShareValidUsers = append(newShareValidUsers, validUser)
  324. }
  325. }
  326. targetShare.ValidUsers = newShareValidUsers
  327. //Create the share
  328. err = s.CreateNewSambaShare(targetShare)
  329. if err != nil {
  330. return err
  331. }
  332. return nil
  333. }
  334. // Get a share by name
  335. func (s *ShareManager) GetShareByName(shareName string) (*ShareConfig, error) {
  336. allShares, err := s.ReadSambaShares()
  337. if err != nil {
  338. return nil, err
  339. }
  340. for _, thisShare := range allShares {
  341. if strings.EqualFold(shareName, thisShare.Name) {
  342. return &thisShare, nil
  343. }
  344. }
  345. return nil, errors.New("target share not found")
  346. }