samba.go 11 KB

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