startsubsys.go 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193
  1. package cmd
  2. import (
  3. "io"
  4. "os"
  5. "os/user"
  6. "path/filepath"
  7. "github.com/rs/xid"
  8. "github.com/rs/zerolog"
  9. "github.com/spf13/cobra"
  10. "github.com/spf13/viper"
  11. "github.com/drakkan/sftpgo/v2/common"
  12. "github.com/drakkan/sftpgo/v2/config"
  13. "github.com/drakkan/sftpgo/v2/dataprovider"
  14. "github.com/drakkan/sftpgo/v2/logger"
  15. "github.com/drakkan/sftpgo/v2/plugin"
  16. "github.com/drakkan/sftpgo/v2/sftpd"
  17. "github.com/drakkan/sftpgo/v2/version"
  18. )
  19. var (
  20. logJournalD = false
  21. preserveHomeDir = false
  22. baseHomeDir = ""
  23. subsystemCmd = &cobra.Command{
  24. Use: "startsubsys",
  25. Short: "Use sftpgo as SFTP file transfer subsystem",
  26. Long: `In this mode SFTPGo speaks the server side of SFTP protocol to stdout and
  27. expects client requests from stdin.
  28. This mode is not intended to be called directly, but from sshd using the
  29. Subsystem option.
  30. For example adding a line like this one in "/etc/ssh/sshd_config":
  31. Subsystem sftp sftpgo startsubsys
  32. Command-line flags should be specified in the Subsystem declaration.
  33. `,
  34. Run: func(cmd *cobra.Command, args []string) {
  35. logSender := "startsubsys"
  36. connectionID := xid.New().String()
  37. logLevel := zerolog.DebugLevel
  38. if !logVerbose {
  39. logLevel = zerolog.InfoLevel
  40. }
  41. logger.SetLogTime(logUTCTime)
  42. if logJournalD {
  43. logger.InitJournalDLogger(logLevel)
  44. } else {
  45. logger.InitStdErrLogger(logLevel)
  46. }
  47. osUser, err := user.Current()
  48. if err != nil {
  49. logger.Error(logSender, connectionID, "unable to get the current user: %v", err)
  50. os.Exit(1)
  51. }
  52. username := osUser.Username
  53. homedir := osUser.HomeDir
  54. logger.Info(logSender, connectionID, "starting SFTPGo %v as subsystem, user %#v home dir %#v config dir %#v base home dir %#v",
  55. version.Get(), username, homedir, configDir, baseHomeDir)
  56. err = config.LoadConfig(configDir, configFile)
  57. if err != nil {
  58. logger.Error(logSender, connectionID, "unable to load configuration: %v", err)
  59. os.Exit(1)
  60. }
  61. dataProviderConf := config.GetProviderConf()
  62. commonConfig := config.GetCommonConfig()
  63. // idle connection are managed externally
  64. commonConfig.IdleTimeout = 0
  65. config.SetCommonConfig(commonConfig)
  66. if err := common.Initialize(config.GetCommonConfig(), dataProviderConf.GetShared()); err != nil {
  67. logger.Error(logSender, connectionID, "%v", err)
  68. os.Exit(1)
  69. }
  70. kmsConfig := config.GetKMSConfig()
  71. if err := kmsConfig.Initialize(); err != nil {
  72. logger.Error(logSender, connectionID, "unable to initialize KMS: %v", err)
  73. os.Exit(1)
  74. }
  75. mfaConfig := config.GetMFAConfig()
  76. err = mfaConfig.Initialize()
  77. if err != nil {
  78. logger.Error(logSender, "", "unable to initialize MFA: %v", err)
  79. os.Exit(1)
  80. }
  81. if err := plugin.Initialize(config.GetPluginsConfig(), logVerbose); err != nil {
  82. logger.Error(logSender, connectionID, "unable to initialize plugin system: %v", err)
  83. os.Exit(1)
  84. }
  85. smtpConfig := config.GetSMTPConfig()
  86. err = smtpConfig.Initialize(configDir)
  87. if err != nil {
  88. logger.Error(logSender, connectionID, "unable to initialize SMTP configuration: %v", err)
  89. os.Exit(1)
  90. }
  91. if dataProviderConf.Driver == dataprovider.SQLiteDataProviderName || dataProviderConf.Driver == dataprovider.BoltDataProviderName {
  92. logger.Debug(logSender, connectionID, "data provider %#v not supported in subsystem mode, using %#v provider",
  93. dataProviderConf.Driver, dataprovider.MemoryDataProviderName)
  94. dataProviderConf.Driver = dataprovider.MemoryDataProviderName
  95. dataProviderConf.Name = ""
  96. dataProviderConf.PreferDatabaseCredentials = true
  97. }
  98. config.SetProviderConf(dataProviderConf)
  99. err = dataprovider.Initialize(dataProviderConf, configDir, false)
  100. if err != nil {
  101. logger.Error(logSender, connectionID, "unable to initialize the data provider: %v", err)
  102. os.Exit(1)
  103. }
  104. httpConfig := config.GetHTTPConfig()
  105. if err := httpConfig.Initialize(configDir); err != nil {
  106. logger.Error(logSender, connectionID, "unable to initialize http client: %v", err)
  107. os.Exit(1)
  108. }
  109. user, err := dataprovider.UserExists(username)
  110. if err == nil {
  111. if user.HomeDir != filepath.Clean(homedir) && !preserveHomeDir {
  112. // update the user
  113. user.HomeDir = filepath.Clean(homedir)
  114. err = dataprovider.UpdateUser(&user, dataprovider.ActionExecutorSystem, "")
  115. if err != nil {
  116. logger.Error(logSender, connectionID, "unable to update user %#v: %v", username, err)
  117. os.Exit(1)
  118. }
  119. }
  120. } else {
  121. user.Username = username
  122. if baseHomeDir != "" && filepath.IsAbs(baseHomeDir) {
  123. user.HomeDir = filepath.Join(baseHomeDir, username)
  124. } else {
  125. user.HomeDir = filepath.Clean(homedir)
  126. }
  127. logger.Debug(logSender, connectionID, "home dir for new user %#v", user.HomeDir)
  128. user.Password = connectionID
  129. user.Permissions = make(map[string][]string)
  130. user.Permissions["/"] = []string{dataprovider.PermAny}
  131. err = dataprovider.AddUser(&user, dataprovider.ActionExecutorSystem, "")
  132. if err != nil {
  133. logger.Error(logSender, connectionID, "unable to add user %#v: %v", username, err)
  134. os.Exit(1)
  135. }
  136. }
  137. err = sftpd.ServeSubSystemConnection(&user, connectionID, os.Stdin, os.Stdout)
  138. if err != nil && err != io.EOF {
  139. logger.Warn(logSender, connectionID, "serving subsystem finished with error: %v", err)
  140. os.Exit(1)
  141. }
  142. logger.Info(logSender, connectionID, "serving subsystem finished")
  143. plugin.Handler.Cleanup()
  144. os.Exit(0)
  145. },
  146. }
  147. )
  148. func init() {
  149. subsystemCmd.Flags().BoolVarP(&preserveHomeDir, "preserve-home", "p", false, `If the user already exists, the existing home
  150. directory will not be changed`)
  151. subsystemCmd.Flags().StringVarP(&baseHomeDir, "base-home-dir", "d", "", `If the user does not exist specify an alternate
  152. starting directory. The home directory for a new
  153. user will be:
  154. [base-home-dir]/[username]
  155. base-home-dir must be an absolute path.`)
  156. subsystemCmd.Flags().BoolVarP(&logJournalD, "log-to-journald", "j", false, `Send logs to journald. Only available on Linux.
  157. Use:
  158. $ journalctl -o verbose -f
  159. To see full logs.
  160. If not set, the logs will be sent to the standard
  161. error`)
  162. addConfigFlags(subsystemCmd)
  163. viper.SetDefault(logVerboseKey, defaultLogVerbose)
  164. viper.BindEnv(logVerboseKey, "SFTPGO_LOG_VERBOSE") //nolint:errcheck
  165. subsystemCmd.Flags().BoolVarP(&logVerbose, logVerboseFlag, "v", viper.GetBool(logVerboseKey),
  166. `Enable verbose logs. This flag can be set
  167. using SFTPGO_LOG_VERBOSE env var too.
  168. `)
  169. viper.BindPFlag(logVerboseKey, subsystemCmd.Flags().Lookup(logVerboseFlag)) //nolint:errcheck
  170. viper.SetDefault(logUTCTimeKey, defaultLogUTCTime)
  171. viper.BindEnv(logUTCTimeKey, "SFTPGO_LOG_UTC_TIME") //nolint:errcheck
  172. subsystemCmd.Flags().BoolVar(&logUTCTime, logUTCTimeFlag, viper.GetBool(logUTCTimeKey),
  173. `Use UTC time for logging. This flag can be set
  174. using SFTPGO_LOG_UTC_TIME env var too.
  175. `)
  176. viper.BindPFlag(logUTCTimeKey, subsystemCmd.Flags().Lookup(logUTCTimeFlag)) //nolint:errcheck
  177. rootCmd.AddCommand(subsystemCmd)
  178. }