123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458 |
- package samba
- import (
- "bufio"
- "errors"
- "fmt"
- "io"
- "os"
- "path/filepath"
- "runtime"
- "strings"
- "time"
- "imuslab.com/arozos/mod/user"
- "imuslab.com/arozos/mod/utils"
- )
- /*
- Samba Share Warpper
- Note that this module only provide exposing of local disk / storage.
- This module do not handle providing the virtualized interface for samba
- */
- type ShareManager struct {
- SambaConfigPath string //Config file for samba, aka smb.conf
- BackupDir string //Backup directory for restoring previous config
- UserHandler *user.UserHandler
- }
- type ShareConfig struct {
- Name string
- Path string
- ValidUsers []string
- ReadOnly bool
- Browseable bool
- GuestOk bool
- }
- func NewSambaShareManager(userHandler *user.UserHandler) (*ShareManager, error) {
- if runtime.GOOS == "linux" {
- //Check if samba installed
- if !utils.FileExists("/bin/smbcontrol") {
- return nil, errors.New("samba not installed")
- }
- } else {
- return nil, errors.New("platform not supported")
- }
- return &ShareManager{
- SambaConfigPath: "/etc/samba/smb.conf",
- BackupDir: "./backup",
- UserHandler: userHandler,
- }, nil
- }
- // ReadSambaShares reads / lists the smb.conf file and extracts all existing shares
- func (s *ShareManager) ReadSambaShares() ([]ShareConfig, error) {
- file, err := os.Open(s.SambaConfigPath)
- if err != nil {
- return nil, err
- }
- defer file.Close()
- var shares []ShareConfig
- var currentShare *ShareConfig
- scanner := bufio.NewScanner(file)
- for scanner.Scan() {
- line := strings.TrimSpace(scanner.Text())
- // Check for section headers
- if strings.HasPrefix(line, "[") && strings.HasSuffix(line, "]") {
- if currentShare != nil {
- shares = append(shares, *currentShare)
- }
- currentShare = &ShareConfig{
- Name: strings.Trim(line, "[]"),
- }
- continue
- }
- // Check if we are currently processing a share section
- if currentShare != nil {
- tokens := strings.SplitN(line, "=", 2)
- if len(tokens) != 2 {
- continue
- }
- key := strings.TrimSpace(tokens[0])
- value := strings.TrimSpace(tokens[1])
- switch key {
- case "path":
- currentShare.Path = value
- case "valid users":
- currentShare.ValidUsers = strings.Fields(value)
- case "read only":
- currentShare.ReadOnly = (value == "yes")
- case "browseable":
- currentShare.Browseable = (value == "yes")
- case "guest ok":
- currentShare.GuestOk = (value == "yes")
- }
- }
- }
- // Add the last share if there is one
- if currentShare != nil {
- shares = append(shares, *currentShare)
- }
- // Check for scanner errors
- if err := scanner.Err(); err != nil {
- return nil, err
- }
- return shares, nil
- }
- // Check if a share name is already used
- func (s *ShareManager) ShareNameExists(sharename string) (bool, error) {
- allShares, err := s.ReadSambaShares()
- if err != nil {
- return false, err
- }
- for _, share := range allShares {
- if strings.EqualFold(share.Name, sharename) {
- return true, nil
- }
- }
- return false, nil
- }
- // A basic filter to remove system created smb shares entry in the list
- func (s *ShareManager) FilterSystemCreatedShares(shares []ShareConfig) []ShareConfig {
- namesToRemove := []string{"global", "homes", "printers", "print$"}
- filteredShares := []ShareConfig{}
- for _, share := range shares {
- if !utils.StringInArray(namesToRemove, share.Name) {
- filteredShares = append(filteredShares, share)
- }
- }
- return filteredShares
- }
- // CreateNewSambaShare converts the shareConfig to string and appends it to smb.conf if the share name does not already exist
- func (s *ShareManager) CreateNewSambaShare(shareToCreate *ShareConfig) error {
- // Path to smb.conf
- smbConfPath := s.SambaConfigPath
- // Open the smb.conf file for reading
- file, err := os.Open(smbConfPath)
- if err != nil {
- return fmt.Errorf("failed to open smb.conf: %v", err)
- }
- defer file.Close()
- // Check if the share already exists
- scanner := bufio.NewScanner(file)
- shareExists := false
- shareNameSection := fmt.Sprintf("[%s]", shareToCreate.Name)
- for scanner.Scan() {
- if strings.EqualFold(strings.TrimSpace(scanner.Text()), shareNameSection) {
- shareExists = true
- break
- }
- }
- if err := scanner.Err(); err != nil {
- return fmt.Errorf("failed to read smb.conf: %v", err)
- }
- if shareExists {
- return fmt.Errorf("share %s already exists", shareToCreate.Name)
- }
- // Convert ShareConfig to string
- shareConfigString := convertShareConfigToString(shareToCreate)
- // Open the smb.conf file for appending
- file, err = os.OpenFile(smbConfPath, os.O_APPEND|os.O_WRONLY, 0644)
- if err != nil {
- return fmt.Errorf("failed to open smb.conf for writing: %v", err)
- }
- defer file.Close()
- // Append the new share configuration
- if _, err := file.WriteString(shareConfigString); err != nil {
- return fmt.Errorf("failed to write to smb.conf: %v", err)
- }
- return nil
- }
- // RemoveSambaShareConfig removes the Samba share configuration from smb.conf
- func (s *ShareManager) RemoveSambaShareConfig(shareName string) error {
- // Check if the share exists
- shareExists, err := s.ShareExists(shareName)
- if err != nil {
- return err
- }
- if !shareExists {
- return errors.New("share not exists")
- }
- //Convert the sharename to correct case for matching
- shareName = s.getCorrectCaseForShareName(shareName)
- // Open the smb.conf file for reading
- file, err := os.Open(s.SambaConfigPath)
- if err != nil {
- return err
- }
- defer file.Close()
- // Create a temporary file to store modified smb.conf
- tmpFile, err := os.CreateTemp("", "smb.conf.*.tmp")
- if err != nil {
- return err
- }
- defer tmpFile.Close()
- // Create a scanner to read the smb.conf file line by line
- scanner := bufio.NewScanner(file)
- for scanner.Scan() {
- line := scanner.Text()
- // Check if the line contains the share name
- if strings.HasPrefix(line, "["+shareName+"]") {
- // Skip the lines until the next section
- for scanner.Scan() {
- checkingLine := scanner.Text()
- if strings.HasPrefix(checkingLine, "[") {
- //The header of the next section is also need to be kept
- fmt.Fprintln(tmpFile, checkingLine)
- break
- }
- }
- continue // Skip writing the share configuration to the temporary file
- }
- // Write the line to the temporary file
- _, err := fmt.Fprintln(tmpFile, line)
- if err != nil {
- return err
- }
- }
- // Check for scanner errors
- if err := scanner.Err(); err != nil {
- return err
- }
- // Close the original smb.conf file
- if err := file.Close(); err != nil {
- return err
- }
- // Close the temporary file
- if err := tmpFile.Close(); err != nil {
- return err
- }
- // Replace the original smb.conf file with the temporary file
- if err := os.Rename(tmpFile.Name(), "/etc/samba/smb.conf"); err != nil {
- return err
- }
- return nil
- }
- // ShareExists checks if a given share name exists in smb.conf
- func (s *ShareManager) ShareExists(shareName string) (bool, error) {
- // Path to smb.conf
- smbConfPath := s.SambaConfigPath
- // Open the smb.conf file for reading
- file, err := os.Open(smbConfPath)
- if err != nil {
- return false, fmt.Errorf("failed to open smb.conf: %v", err)
- }
- defer file.Close()
- // Check if the share already exists
- scanner := bufio.NewScanner(file)
- shareNameSection := fmt.Sprintf("[%s]", shareName)
- for scanner.Scan() {
- if strings.EqualFold(shareNameSection, strings.TrimSpace(scanner.Text())) {
- return true, nil
- }
- }
- if err := scanner.Err(); err != nil {
- return false, fmt.Errorf("failed to read smb.conf: %v", err)
- }
- return false, nil
- }
- // Backup the current smb.conf to the backup folder
- func (s *ShareManager) BackupSmbConf() error {
- // Define source and backup directory
- sourceFile := s.SambaConfigPath
- backupDir := s.BackupDir
- // Ensure the backup directory exists
- err := os.MkdirAll(backupDir, 0755)
- if err != nil {
- return fmt.Errorf("failed to create backup directory: %v", err)
- }
- // Create a timestamped backup filename
- timestamp := time.Now().Format("20060102_150405")
- backupFile := filepath.Join(backupDir, fmt.Sprintf("%s.smb.conf", timestamp))
- // Open the source file
- src, err := os.Open(sourceFile)
- if err != nil {
- return fmt.Errorf("failed to open source file: %v", err)
- }
- defer src.Close()
- // Create the destination file
- dst, err := os.Create(backupFile)
- if err != nil {
- return fmt.Errorf("failed to create backup file: %v", err)
- }
- defer dst.Close()
- // Copy the contents of the source file to the backup file
- _, err = io.Copy(dst, src)
- if err != nil {
- return fmt.Errorf("failed to copy file contents: %v", err)
- }
- return nil
- }
- // Add a new user to smb.conf share by name
- func (s *ShareManager) AddUserToSambaShare(shareName, username string) error {
- targetShare, err := s.GetShareByName(shareName)
- if err != nil {
- return err
- }
- if s.UserCanAccessShare(targetShare, username) {
- //User already have no access to this share, no need to delete
- return nil
- }
- //User not in this share. Append user name in this valid users
- targetShare.ValidUsers = append(targetShare.ValidUsers, username)
- //Delete the old one and create the new share with updated user list
- err = s.RemoveSambaShareConfig(shareName)
- if err != nil {
- return err
- }
- err = s.CreateNewSambaShare(targetShare)
- if err != nil {
- return err
- }
- return nil
- }
- // Return if the user can access this samba share
- func (s *ShareManager) UserCanAccessShare(targetSmbShare *ShareConfig, username string) bool {
- return utils.StringInArray(targetSmbShare.ValidUsers, username)
- }
- // Get a list of shares that this user have access to
- func (s *ShareManager) GetUsersShare(username string) ([]*ShareConfig, error) {
- allShares, err := s.ReadSambaShares()
- if err != nil {
- return nil, err
- }
- userAccessibleShares := []*ShareConfig{}
- for _, thisShare := range allShares {
- if s.UserCanAccessShare(&thisShare, username) {
- thisShareObject := thisShare
- userAccessibleShares = append(userAccessibleShares, &thisShareObject)
- }
- }
- return userAccessibleShares, nil
- }
- // Remove a user from smb.conf share by name
- func (s *ShareManager) RemoveUserFromSambaShare(shareName, username string) error {
- targetShare, err := s.GetShareByName(shareName)
- if err != nil {
- return err
- }
- if !s.UserCanAccessShare(targetShare, username) {
- //User already have no access to this share, no need to delete
- return nil
- }
- if len(targetShare.ValidUsers) == 1 && strings.EqualFold(targetShare.ValidUsers[0], username) {
- //This user is the only person who can access this share
- //Remove the share entirely
- //Delete the old one and create the new share with updated user list
- err = s.RemoveSambaShareConfig(shareName)
- if err != nil {
- return err
- }
- return nil
- }
- //User is in this share but this share contain other users.
- err = s.RemoveSambaShareConfig(shareName)
- if err != nil {
- return err
- }
- //Create a new valid user list
- newShareValidUsers := []string{}
- for _, validUser := range targetShare.ValidUsers {
- if !strings.EqualFold(validUser, username) {
- newShareValidUsers = append(newShareValidUsers, validUser)
- }
- }
- targetShare.ValidUsers = newShareValidUsers
- //Create the share
- err = s.CreateNewSambaShare(targetShare)
- if err != nil {
- return err
- }
- return nil
- }
- // Get a share by name
- func (s *ShareManager) GetShareByName(shareName string) (*ShareConfig, error) {
- allShares, err := s.ReadSambaShares()
- if err != nil {
- return nil, err
- }
- for _, thisShare := range allShares {
- if strings.EqualFold(shareName, thisShare.Name) {
- return &thisShare, nil
- }
- }
- return nil, errors.New("target share not found")
- }
|