diff --git a/.github/dependabot.yml b/.github/dependabot.yml index d0ad1de..1141609 100644 --- a/.github/dependabot.yml +++ b/.github/dependabot.yml @@ -6,8 +6,8 @@ updates: target-branch: "develop" schedule: interval: "daily" - timezone: "Europe/London" - time: "03:00" + timezone: "Etc/UTC" + time: "07:00" labels: - "PR • MAINTENANCE" assignees: @@ -26,8 +26,8 @@ updates: target-branch: "develop" schedule: interval: "daily" - timezone: "Europe/London" - time: "04:00" + timezone: "Etc/UTC" + time: "08:00" labels: - "PR • MAINTENANCE" assignees: diff --git a/.github/images/usage-container.svg b/.github/images/usage-container.svg index 506b518..3eba0d6 100644 --- a/.github/images/usage-container.svg +++ b/.github/images/usage-container.svg @@ -1,5 +1,5 @@ - - + + Atlassian Cloud Backuper Usage @@ -68,9 +68,9 @@ } - - - + + + @@ -83,44 +83,46 @@ --config, -c file ........................ Path to configuration file --interactive, -I ........................ Interactive mode --server, -S ............................. Server mode - --no-color, -nc .......................... Disable colors in output - --help, -h ............................... Show this help message - --version, -v ............................ Show version - --access-account name .................... Account name (ACCESS_ACCOUNT) - --access-email email ..................... User email with access to API (ACCESS_EMAIL) - --access-api-key key ..................... API key (ACCESS_API_KEY) - --server-ip ip ........................... HTTP server IP (SERVER_IP) - --server-port port ....................... HTTP server port (SERVER_PORT) - --server-access-token token .............. HTTP access token (SERVER_ACCESS_TOKEN) - --storage-type fs/sftp/s3 ................ Storage type (STORAGE_TYPE) - --storage-encryption-key key ............. Data encryption key (STORAGE_ENCRYPTION_KEY) - --storage-fs-path path ................... Path on system for backups (STORAGE_FS_PATH) - --storage-fs-mode mode ................... File mode on system (STORAGE_FS_MODE) - --storage-sftp-host host ................. SFTP host (STORAGE_SFTP_HOST) - --storage-sftp-user name ................. SFTP user name (STORAGE_SFTP_USER) - --storage-sftp-key key ................... Base64-encoded private key (STORAGE_SFTP_KEY) - --storage-sftp-path path ................. Path on SFTP (STORAGE_SFTP_PATH) - --storage-sftp-mode mode ................. File mode on SFTP (STORAGE_SFTP_MODE) - --storage-s3-host host ................... S3 host (STORAGE_S3_HOST) - --storage-s3-region region ............... S3 region (STORAGE_S3_REGION) - --storage-s3-access-key id ............... S3 access key ID (STORAGE_S3_ACCESS_KEY) - --storage-s3-secret-key key .............. S3 access secret key (STORAGE_S3_SECRET_KEY) - --storage-s3-bucket name ................. S3 bucket (STORAGE_S3_BUCKET) - --storage-s3-path path ................... Path for backups (STORAGE_S3_PATH) - --jira-output-file template .............. Jira backup output file name template - (JIRA_OUTPUT_FILE) - --jira-include-attachments yes/no ........ Include attachments to Jira backup - (JIRA_INCLUDE_ATTACHMENTS) - --jira-cloud-format yes/no ............... Create Jira backup for Cloud (JIRA_CLOUD_FORMAT) - --confluence-output-file template ........ Confluence backup output file name template - (CONFLUENCE_OUTPUT_FILE) - --confluence-include-attachments yes/no .. Include attachments to Confluence backup - (CONFLUENCE_INCLUDE_ATTACHMENTS) - --confluence-cloud-format yes/no ......... Create Confluence backup for Cloud - (CONFLUENCE_CLOUD_FORMAT) - --temp-dir path .......................... Path to directory for temporary data (TEMP_DIR) - --log-format text/json ................... Log format (LOG_FORMAT) - --log-level level ........................ Log level (LOG_LEVEL) + --force, -F .............................. Force backup generation + --no-color, -nc .......................... Disable colors in output + --help, -h ............................... Show this help message + --version, -v ............................ Show version + --access-account name .................... Account name (ACCESS_ACCOUNT) + --access-email email ..................... User email with access to API (ACCESS_EMAIL) + --access-api-key key ..................... API key (ACCESS_API_KEY) + --server-ip ip ........................... HTTP server IP (SERVER_IP) + --server-port port ....................... HTTP server port (SERVER_PORT) + --server-access-token token .............. HTTP access token (SERVER_ACCESS_TOKEN) + --storage-type fs/sftp/s3 ................ Storage type (STORAGE_TYPE) + --storage-encryption-key key ............. Data encryption key (STORAGE_ENCRYPTION_KEY) + --storage-fs-path path ................... Path on system for backups (STORAGE_FS_PATH) + --storage-fs-mode mode ................... File mode on system (STORAGE_FS_MODE) + --storage-sftp-host host ................. SFTP host (STORAGE_SFTP_HOST) + --storage-sftp-user name ................. SFTP user name (STORAGE_SFTP_USER) + --storage-sftp-key key ................... Base64-encoded private key (STORAGE_SFTP_KEY) + --storage-sftp-path path ................. Path on SFTP (STORAGE_SFTP_PATH) + --storage-sftp-mode mode ................. File mode on SFTP (STORAGE_SFTP_MODE) + --storage-s3-host host ................... S3 host (STORAGE_S3_HOST) + --storage-s3-region region ............... S3 region (STORAGE_S3_REGION) + --storage-s3-access-key id ............... S3 access key ID (STORAGE_S3_ACCESS_KEY) + --storage-s3-secret-key key .............. S3 access secret key (STORAGE_S3_SECRET_KEY) + --storage-s3-bucket name ................. S3 bucket (STORAGE_S3_BUCKET) + --storage-s3-path path ................... Path for backups (STORAGE_S3_PATH) + --storage-s3-part-size num ............... Uploading part size (in MB) (STORAGE_S3_PART_SIZE) + --jira-output-file template .............. Jira backup output file name template + (JIRA_OUTPUT_FILE) + --jira-include-attachments yes/no ........ Include attachments to Jira backup + (JIRA_INCLUDE_ATTACHMENTS) + --jira-cloud-format yes/no ............... Create Jira backup for Cloud (JIRA_CLOUD_FORMAT) + --confluence-output-file template ........ Confluence backup output file name template + (CONFLUENCE_OUTPUT_FILE) + --confluence-include-attachments yes/no .. Include attachments to Confluence backup + (CONFLUENCE_INCLUDE_ATTACHMENTS) + --confluence-cloud-format yes/no ......... Create Confluence backup for Cloud + (CONFLUENCE_CLOUD_FORMAT) + --temp-dir path .......................... Path to directory for temporary data (TEMP_DIR) + --log-format text/json ................... Log format (LOG_FORMAT) + --log-level level ........................ Log level (LOG_LEVEL) - + diff --git a/.github/images/usage.svg b/.github/images/usage.svg index 6512674..21c2566 100644 --- a/.github/images/usage.svg +++ b/.github/images/usage.svg @@ -1,5 +1,5 @@ - - + + Atlassian Cloud Backuper Usage @@ -68,7 +68,7 @@ } - + @@ -83,9 +83,10 @@ --config, -c file .. Path to configuration file --interactive, -I .. Interactive mode --server, -S ....... Server mode - --no-color, -nc .... Disable colors in output - --help, -h ......... Show this help message - --version, -v ...... Show version + --force, -F ........ Force backup generation + --no-color, -nc .... Disable colors in output + --help, -h ......... Show this help message + --version, -v ...... Show version - + diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index 3d1f55c..dfcbd25 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -15,7 +15,7 @@ jobs: strategy: matrix: - go: [ '1.21.x', '1.22.x' ] + go: [ '1.22.x', '1.23.x' ] steps: - name: Checkout diff --git a/Makefile b/Makefile index 02c3bad..4b23501 100644 --- a/Makefile +++ b/Makefile @@ -1,6 +1,6 @@ ################################################################################ -# This Makefile generated by GoMakeGen 3.0.5 using next command: +# This Makefile generated by GoMakeGen 3.1.0 using next command: # gomakegen --mod . # # More info: https://kaos.sh/gomakegen @@ -11,6 +11,10 @@ ifdef VERBOSE ## Print verbose information (Flag) VERBOSE_FLAG = -v endif +ifdef PROXY ## Force proxy usage for downloading dependencies (Flag) +export GOPROXY=https://proxy.golang.org/cached-only,direct +endif + COMPAT ?= 1.19 MAKEDIR = $(dir $(realpath $(firstword $(MAKEFILE_LIST)))) GITREV ?= $(shell test -s $(MAKEDIR)/.git && git rev-parse --short HEAD) @@ -109,6 +113,6 @@ help: ## Show this info | sed 's/ifdef //' \ | awk 'BEGIN {FS = " .*?## "}; {printf " \033[32m%-11s\033[0m %s\n", $$1, $$2}' @echo -e '' - @echo -e '\033[90mGenerated by GoMakeGen 3.0.5\033[0m\n' + @echo -e '\033[90mGenerated by GoMakeGen 3.1.0\033[0m\n' ################################################################################ diff --git a/README.md b/README.md index db76463..e4f882f 100644 --- a/README.md +++ b/README.md @@ -2,6 +2,7 @@

Code Climate Maintainability + Codacy badge GitHub Actions CI Status GitHub Actions CodeQL Status diff --git a/app/app.go b/app/app.go index e95231c..3348382 100644 --- a/app/app.go +++ b/app/app.go @@ -42,7 +42,7 @@ import ( // Basic utility info const ( APP = "Atlassian Cloud Backuper" - VER = "0.1.0" + VER = "0.2.0" DESC = "Tool for backuping Atlassian cloud services (Jira and Confluence)" ) @@ -53,6 +53,7 @@ const ( OPT_CONFIG = "c:config" OPT_INTERACTIVE = "I:interactive" OPT_SERVER = "S:server" + OPT_FORCE = "F:force" OPT_NO_COLOR = "nc:no-color" OPT_HELP = "h:help" OPT_VER = "v:version" @@ -84,6 +85,7 @@ const ( STORAGE_S3_SECRET_KEY = "storage-s3:secret-key" STORAGE_S3_BUCKET = "storage-s3:bucket" STORAGE_S3_PATH = "storage-s3:path" + STORAGE_S3_PART_SIZE = "storage-s3:part-size" JIRA_OUTPUT_FILE = "jira:output-file" JIRA_INCLUDE_ATTACHMENTS = "jira:include-attachments" JIRA_CLOUD_FORMAT = "jira:cloud-format" @@ -116,6 +118,7 @@ const ( // optMap contains information about all supported options var optMap = options.Map{ OPT_CONFIG: {Value: "/etc/atlassian-cloud-backuper.knf"}, + OPT_FORCE: {Type: options.BOOL}, OPT_INTERACTIVE: {Type: options.BOOL}, OPT_SERVER: {Type: options.BOOL}, OPT_NO_COLOR: {Type: options.BOOL}, @@ -249,7 +252,7 @@ func addExtraOptions(m options.Map) { STORAGE_SFTP_HOST, STORAGE_SFTP_USER, STORAGE_SFTP_KEY, STORAGE_SFTP_PATH, STORAGE_SFTP_MODE, STORAGE_S3_HOST, STORAGE_S3_ACCESS_KEY, STORAGE_S3_SECRET_KEY, - STORAGE_S3_BUCKET, STORAGE_S3_PATH, + STORAGE_S3_BUCKET, STORAGE_S3_PATH, STORAGE_S3_PART_SIZE, JIRA_OUTPUT_FILE, JIRA_INCLUDE_ATTACHMENTS, JIRA_CLOUD_FORMAT, CONFLUENCE_OUTPUT_FILE, CONFLUENCE_INCLUDE_ATTACHMENTS, CONFLUENCE_CLOUD_FORMAT, TEMP_DIR, @@ -284,7 +287,7 @@ func loadConfig() error { STORAGE_SFTP_HOST, STORAGE_SFTP_USER, STORAGE_SFTP_KEY, STORAGE_SFTP_PATH, STORAGE_SFTP_MODE, STORAGE_S3_HOST, STORAGE_S3_REGION, STORAGE_S3_ACCESS_KEY, - STORAGE_S3_SECRET_KEY, STORAGE_S3_BUCKET, STORAGE_S3_PATH, + STORAGE_S3_SECRET_KEY, STORAGE_S3_BUCKET, STORAGE_S3_PATH, STORAGE_S3_PART_SIZE, JIRA_OUTPUT_FILE, JIRA_INCLUDE_ATTACHMENTS, JIRA_CLOUD_FORMAT, CONFLUENCE_OUTPUT_FILE, CONFLUENCE_INCLUDE_ATTACHMENTS, CONFLUENCE_CLOUD_FORMAT, TEMP_DIR, @@ -334,6 +337,8 @@ func validateConfig() error { &knf.Validator{STORAGE_S3_ACCESS_KEY, knfv.Set, nil}, &knf.Validator{STORAGE_S3_SECRET_KEY, knfv.Set, nil}, &knf.Validator{STORAGE_S3_BUCKET, knfv.Set, nil}, + &knf.Validator{STORAGE_S3_PART_SIZE, knfv.Greater, 5}, + &knf.Validator{STORAGE_S3_PART_SIZE, knfv.Less, 5_000}, ) } @@ -498,6 +503,7 @@ func genUsage(section string) *usage.Info { info.AddOption(OPT_CONFIG, "Path to configuration file", "file") info.AddOption(OPT_INTERACTIVE, "Interactive mode") info.AddOption(OPT_SERVER, "Server mode") + info.AddOption(OPT_FORCE, "Force backup generation") info.AddOption(OPT_NO_COLOR, "Disable colors in output") info.AddOption(OPT_HELP, "Show this help message") info.AddOption(OPT_VER, "Show version") @@ -524,6 +530,7 @@ func genUsage(section string) *usage.Info { addUnitedOption(info, STORAGE_S3_SECRET_KEY, "S3 access secret key", "key") addUnitedOption(info, STORAGE_S3_BUCKET, "S3 bucket", "name") addUnitedOption(info, STORAGE_S3_PATH, "Path for backups", "path") + addUnitedOption(info, STORAGE_S3_PART_SIZE, "Uploading part size (in MB)", "num") addUnitedOption(info, JIRA_OUTPUT_FILE, "Jira backup output file name template", "template") addUnitedOption(info, JIRA_INCLUDE_ATTACHMENTS, "Include attachments to Jira backup", "yes/no") addUnitedOption(info, JIRA_CLOUD_FORMAT, "Create Jira backup for Cloud", "yes/no") diff --git a/app/basic.go b/app/basic.go index 876078b..9d15704 100644 --- a/app/basic.go +++ b/app/basic.go @@ -73,7 +73,7 @@ func startApp(args options.Arguments) error { tmpFile := path.Join(tmpDir, outputFileName) - err = bkpr.Backup(tmpFile) + err = bkpr.Backup(tmpFile, options.GetB(OPT_FORCE)) if err != nil { spinner.Done(false) diff --git a/app/common.go b/app/common.go index 4fe873a..d122a45 100644 --- a/app/common.go +++ b/app/common.go @@ -104,9 +104,9 @@ func getUploader(target string) (uploader.Uploader, error) { switch strings.ToLower(knfu.GetS(STORAGE_TYPE)) { case STORAGE_FS: return fs.NewUploader(&fs.Config{ + Secret: secret, Path: path.Join(knfu.GetS(STORAGE_FS_PATH), target), Mode: knfu.GetM(STORAGE_FS_MODE, 0600), - Secret: secret, }) case STORAGE_SFTP: @@ -117,23 +117,24 @@ func getUploader(target string) (uploader.Uploader, error) { } return sftp.NewUploader(&sftp.Config{ + Secret: secret, Host: knfu.GetS(STORAGE_SFTP_HOST), User: knfu.GetS(STORAGE_SFTP_USER), Key: keyData, Path: path.Join(knfu.GetS(STORAGE_SFTP_PATH), target), Mode: knfu.GetM(STORAGE_SFTP_MODE, 0600), - Secret: secret, }) case STORAGE_S3: return s3.NewUploader(&s3.Config{ + Secret: secret, Host: knfu.GetS(STORAGE_S3_HOST), Region: knfu.GetS(STORAGE_S3_REGION), AccessKeyID: knfu.GetS(STORAGE_S3_ACCESS_KEY), SecretKey: knfu.GetS(STORAGE_S3_SECRET_KEY), Bucket: knfu.GetS(STORAGE_S3_BUCKET), Path: path.Join(knfu.GetS(STORAGE_S3_PATH), target), - Secret: secret, + PartSize: knfu.GetI64(STORAGE_S3_PART_SIZE, 32), }) } diff --git a/app/server.go b/app/server.go index 8d25121..194816d 100644 --- a/app/server.go +++ b/app/server.go @@ -58,6 +58,7 @@ func createBackupHandler(rw http.ResponseWriter, r *http.Request) { target := strings.ToLower(r.URL.Query().Get("target")) token := r.URL.Query().Get("token") + force := r.URL.Query().Get("force") != "" err := validateRequestQuery(target, token) @@ -75,7 +76,7 @@ func createBackupHandler(rw http.ResponseWriter, r *http.Request) { return } - taskID, err := bkpr.Start() + taskID, err := bkpr.Start(force) if err != nil { log.Error("Can't create backup: %v", err) diff --git a/backuper/backuper.go b/backuper/backuper.go index a7f8821..0d34413 100644 --- a/backuper/backuper.go +++ b/backuper/backuper.go @@ -28,13 +28,13 @@ const ( // Backuper is generic backuper interface type Backuper interface { // Backup runs backup process - Backup(outputFile string) error + Backup(outputFile string, force bool) error // SetDispatcher sets events dispatcher SetDispatcher(d *events.Dispatcher) // Start creates task for backuping data - Start() (string, error) + Start(force bool) (string, error) // Progress monitors backup creation progress Progress(taskID string) (string, error) diff --git a/backuper/confluence/confluence-backuper.go b/backuper/confluence/confluence-backuper.go index e445c70..5888205 100644 --- a/backuper/confluence/confluence-backuper.go +++ b/backuper/confluence/confluence-backuper.go @@ -74,8 +74,8 @@ func (b *ConfluenceBackuper) SetDispatcher(d *events.Dispatcher) { } // Backup starts backup process -func (b *ConfluenceBackuper) Backup(outputFile string) error { - _, err := b.Start() +func (b *ConfluenceBackuper) Backup(outputFile string, force bool) error { + _, err := b.Start(force) if err != nil { return err @@ -91,13 +91,13 @@ func (b *ConfluenceBackuper) Backup(outputFile string) error { } // Start creates task for backuping data -func (b *ConfluenceBackuper) Start() (string, error) { +func (b *ConfluenceBackuper) Start(force bool) (string, error) { log.Info("Starting Confluence backup process for account %s…", b.config.Account) log.Info("Checking for existing backup task…") info, _ := b.getBackupProgress() - if info != nil && !info.IsOutdated { + if !force && info != nil && !info.IsOutdated { log.Info( "Found previously created backup task", log.F{"backup-status", info.CurrentStatus}, @@ -107,7 +107,11 @@ func (b *ConfluenceBackuper) Start() (string, error) { log.F{"backup-outdated", info.IsOutdated}, ) } else { - log.Info("No previously created backup task or task is outdated, starting new backup…") + if force { + log.Info("Starting new backup…") + } else { + log.Info("No previously created backup task or task is outdated, starting new backup…") + } err := b.startBackup() diff --git a/backuper/jira/jira-backuper.go b/backuper/jira/jira-backuper.go index a189d56..fc5b765 100644 --- a/backuper/jira/jira-backuper.go +++ b/backuper/jira/jira-backuper.go @@ -75,8 +75,8 @@ func (b *JiraBackuper) SetDispatcher(d *events.Dispatcher) { } // Backup starts backup process -func (b *JiraBackuper) Backup(outputFile string) error { - backupTaskID, err := b.Start() +func (b *JiraBackuper) Backup(outputFile string, force bool) error { + backupTaskID, err := b.Start(force) if err != nil { return err @@ -92,20 +92,25 @@ func (b *JiraBackuper) Backup(outputFile string) error { } // Start creates task for backuping data -func (b *JiraBackuper) Start() (string, error) { +func (b *JiraBackuper) Start(force bool) (string, error) { var err error var backupTaskID string log.Info("Starting Jira backup process for account %s…", b.config.Account) - log.Info("Checking for existing backup task…") - backupTaskID, _ = b.getLastTaskID() + if !force { + log.Info("Checking for existing backup task…") - if backupTaskID != "" { - log.Info("Found previously created backup task with ID %s", backupTaskID) + backupTaskID, _ = b.getLastTaskID() + + if backupTaskID == "" { + log.Info("No previously created task found, starting new backup…") + } } else { - log.Info("No previously created task found, starting new backup…") + log.Info("Starting new backup…") + } + if backupTaskID == "" { backupTaskID, err = b.startBackup() if err != nil { diff --git a/common/atlassian-cloud-backuper-container.knf b/common/atlassian-cloud-backuper-container.knf index ea8d82f..e7258f9 100644 --- a/common/atlassian-cloud-backuper-container.knf +++ b/common/atlassian-cloud-backuper-container.knf @@ -73,6 +73,9 @@ # Path to directory with backups path: + # Part size in MB (default: 32) + part-size: 32 + [jira] # Backup file name with date tags (default: jira-backup-%Y-%m-%d.zip) diff --git a/common/atlassian-cloud-backuper.knf b/common/atlassian-cloud-backuper.knf index fa40a4a..7f3dbf2 100644 --- a/common/atlassian-cloud-backuper.knf +++ b/common/atlassian-cloud-backuper.knf @@ -73,6 +73,9 @@ # Path to directory with backups path: + # Part size in MB (default: 32) + part-size: 32 + [jira] # Backup file name with date tags (default: jira-backup-%Y-%m-%d.zip) diff --git a/common/atlassian-cloud-backuper.spec b/common/atlassian-cloud-backuper.spec index b2bef66..a374885 100644 --- a/common/atlassian-cloud-backuper.spec +++ b/common/atlassian-cloud-backuper.spec @@ -10,7 +10,7 @@ Summary: Tool for backuping Atlassian cloud services Name: atlassian-cloud-backuper -Version: 0.1.0 +Version: 0.2.0 Release: 0%{?dist} Group: Applications/System License: Apache License, Version 2.0 @@ -114,7 +114,9 @@ rm -rf %{buildroot} %changelog * Tue Jul 23 2024 Anton Novojilov - 0.1.0-0 +- Added option (-F/--force) and query param (force) to force backup creation - Added data encryption feature +- Added multipart uploading to S3 storage - Added server mode - Code refactoring - Dependencies update diff --git a/go.mod b/go.mod index 7043e3f..26eca94 100644 --- a/go.mod +++ b/go.mod @@ -1,29 +1,30 @@ module github.com/essentialkaos/atlassian-cloud-backuper -go 1.19 +go 1.21 require ( - github.com/aws/aws-sdk-go-v2 v1.30.3 - github.com/aws/aws-sdk-go-v2/credentials v1.17.27 - github.com/aws/aws-sdk-go-v2/service/s3 v1.58.2 - github.com/essentialkaos/ek/v13 v13.2.0 + github.com/aws/aws-sdk-go-v2 v1.30.5 + github.com/aws/aws-sdk-go-v2/credentials v1.17.32 + github.com/aws/aws-sdk-go-v2/feature/s3/manager v1.17.18 + github.com/aws/aws-sdk-go-v2/service/s3 v1.61.2 + github.com/essentialkaos/ek/v13 v13.3.5 github.com/essentialkaos/katana v0.2.0 github.com/pkg/sftp v1.13.6 - golang.org/x/crypto v0.25.0 + golang.org/x/crypto v0.26.0 ) require ( - github.com/aws/aws-sdk-go-v2/aws/protocol/eventstream v1.6.3 // indirect - github.com/aws/aws-sdk-go-v2/internal/configsources v1.3.15 // indirect - github.com/aws/aws-sdk-go-v2/internal/endpoints/v2 v2.6.15 // indirect - github.com/aws/aws-sdk-go-v2/internal/v4a v1.3.15 // indirect - github.com/aws/aws-sdk-go-v2/service/internal/accept-encoding v1.11.3 // indirect - github.com/aws/aws-sdk-go-v2/service/internal/checksum v1.3.17 // indirect - github.com/aws/aws-sdk-go-v2/service/internal/presigned-url v1.11.17 // indirect - github.com/aws/aws-sdk-go-v2/service/internal/s3shared v1.17.15 // indirect - github.com/aws/smithy-go v1.20.3 // indirect + github.com/aws/aws-sdk-go-v2/aws/protocol/eventstream v1.6.4 // indirect + github.com/aws/aws-sdk-go-v2/internal/configsources v1.3.17 // indirect + github.com/aws/aws-sdk-go-v2/internal/endpoints/v2 v2.6.17 // indirect + github.com/aws/aws-sdk-go-v2/internal/v4a v1.3.17 // indirect + github.com/aws/aws-sdk-go-v2/service/internal/accept-encoding v1.11.4 // indirect + github.com/aws/aws-sdk-go-v2/service/internal/checksum v1.3.19 // indirect + github.com/aws/aws-sdk-go-v2/service/internal/presigned-url v1.11.19 // indirect + github.com/aws/aws-sdk-go-v2/service/internal/s3shared v1.17.17 // indirect + github.com/aws/smithy-go v1.20.4 // indirect github.com/essentialkaos/depsy v1.3.0 // indirect github.com/essentialkaos/sio v1.0.0 // indirect github.com/kr/fs v0.1.0 // indirect - golang.org/x/sys v0.22.0 // indirect + golang.org/x/sys v0.24.0 // indirect ) diff --git a/go.sum b/go.sum index b04bb0d..90a5af2 100644 --- a/go.sum +++ b/go.sum @@ -1,35 +1,50 @@ -github.com/aws/aws-sdk-go-v2 v1.30.3 h1:jUeBtG0Ih+ZIFH0F4UkmL9w3cSpaMv9tYYDbzILP8dY= -github.com/aws/aws-sdk-go-v2 v1.30.3/go.mod h1:nIQjQVp5sfpQcTc9mPSr1B0PaWK5ByX9MOoDadSN4lc= -github.com/aws/aws-sdk-go-v2/aws/protocol/eventstream v1.6.3 h1:tW1/Rkad38LA15X4UQtjXZXNKsCgkshC3EbmcUmghTg= -github.com/aws/aws-sdk-go-v2/aws/protocol/eventstream v1.6.3/go.mod h1:UbnqO+zjqk3uIt9yCACHJ9IVNhyhOCnYk8yA19SAWrM= -github.com/aws/aws-sdk-go-v2/credentials v1.17.27 h1:2raNba6gr2IfA0eqqiP2XiQ0UVOpGPgDSi0I9iAP+UI= -github.com/aws/aws-sdk-go-v2/credentials v1.17.27/go.mod h1:gniiwbGahQByxan6YjQUMcW4Aov6bLC3m+evgcoN4r4= -github.com/aws/aws-sdk-go-v2/internal/configsources v1.3.15 h1:SoNJ4RlFEQEbtDcCEt+QG56MY4fm4W8rYirAmq+/DdU= -github.com/aws/aws-sdk-go-v2/internal/configsources v1.3.15/go.mod h1:U9ke74k1n2bf+RIgoX1SXFed1HLs51OgUSs+Ph0KJP8= -github.com/aws/aws-sdk-go-v2/internal/endpoints/v2 v2.6.15 h1:C6WHdGnTDIYETAm5iErQUiVNsclNx9qbJVPIt03B6bI= -github.com/aws/aws-sdk-go-v2/internal/endpoints/v2 v2.6.15/go.mod h1:ZQLZqhcu+JhSrA9/NXRm8SkDvsycE+JkV3WGY41e+IM= -github.com/aws/aws-sdk-go-v2/internal/v4a v1.3.15 h1:Z5r7SycxmSllHYmaAZPpmN8GviDrSGhMS6bldqtXZPw= -github.com/aws/aws-sdk-go-v2/internal/v4a v1.3.15/go.mod h1:CetW7bDE00QoGEmPUoZuRog07SGVAUVW6LFpNP0YfIg= -github.com/aws/aws-sdk-go-v2/service/internal/accept-encoding v1.11.3 h1:dT3MqvGhSoaIhRseqw2I0yH81l7wiR2vjs57O51EAm8= -github.com/aws/aws-sdk-go-v2/service/internal/accept-encoding v1.11.3/go.mod h1:GlAeCkHwugxdHaueRr4nhPuY+WW+gR8UjlcqzPr1SPI= -github.com/aws/aws-sdk-go-v2/service/internal/checksum v1.3.17 h1:YPYe6ZmvUfDDDELqEKtAd6bo8zxhkm+XEFEzQisqUIE= -github.com/aws/aws-sdk-go-v2/service/internal/checksum v1.3.17/go.mod h1:oBtcnYua/CgzCWYN7NZ5j7PotFDaFSUjCYVTtfyn7vw= -github.com/aws/aws-sdk-go-v2/service/internal/presigned-url v1.11.17 h1:HGErhhrxZlQ044RiM+WdoZxp0p+EGM62y3L6pwA4olE= -github.com/aws/aws-sdk-go-v2/service/internal/presigned-url v1.11.17/go.mod h1:RkZEx4l0EHYDJpWppMJ3nD9wZJAa8/0lq9aVC+r2UII= -github.com/aws/aws-sdk-go-v2/service/internal/s3shared v1.17.15 h1:246A4lSTXWJw/rmlQI+TT2OcqeDMKBdyjEQrafMaQdA= -github.com/aws/aws-sdk-go-v2/service/internal/s3shared v1.17.15/go.mod h1:haVfg3761/WF7YPuJOER2MP0k4UAXyHaLclKXB6usDg= -github.com/aws/aws-sdk-go-v2/service/s3 v1.58.2 h1:sZXIzO38GZOU+O0C+INqbH7C2yALwfMWpd64tONS/NE= -github.com/aws/aws-sdk-go-v2/service/s3 v1.58.2/go.mod h1:Lcxzg5rojyVPU/0eFwLtcyTaek/6Mtic5B1gJo7e/zE= -github.com/aws/smithy-go v1.20.3 h1:ryHwveWzPV5BIof6fyDvor6V3iUL7nTfiTKXHiW05nE= -github.com/aws/smithy-go v1.20.3/go.mod h1:krry+ya/rV9RDcV/Q16kpu6ypI4K2czasz0NC3qS14E= +github.com/aws/aws-sdk-go-v2 v1.30.5 h1:mWSRTwQAb0aLE17dSzztCVJWI9+cRMgqebndjwDyK0g= +github.com/aws/aws-sdk-go-v2 v1.30.5/go.mod h1:CT+ZPWXbYrci8chcARI3OmI/qgd+f6WtuLOoaIA8PR0= +github.com/aws/aws-sdk-go-v2/aws/protocol/eventstream v1.6.4 h1:70PVAiL15/aBMh5LThwgXdSQorVr91L127ttckI9QQU= +github.com/aws/aws-sdk-go-v2/aws/protocol/eventstream v1.6.4/go.mod h1:/MQxMqci8tlqDH+pjmoLu1i0tbWCUP1hhyMRuFxpQCw= +github.com/aws/aws-sdk-go-v2/config v1.27.33 h1:Nof9o/MsmH4oa0s2q9a0k7tMz5x/Yj5k06lDODWz3BU= +github.com/aws/aws-sdk-go-v2/config v1.27.33/go.mod h1:kEqdYzRb8dd8Sy2pOdEbExTTF5v7ozEXX0McgPE7xks= +github.com/aws/aws-sdk-go-v2/credentials v1.17.32 h1:7Cxhp/BnT2RcGy4VisJ9miUPecY+lyE9I8JvcZofn9I= +github.com/aws/aws-sdk-go-v2/credentials v1.17.32/go.mod h1:P5/QMF3/DCHbXGEGkdbilXHsyTBX5D3HSwcrSc9p20I= +github.com/aws/aws-sdk-go-v2/feature/ec2/imds v1.16.13 h1:pfQ2sqNpMVK6xz2RbqLEL0GH87JOwSxPV2rzm8Zsb74= +github.com/aws/aws-sdk-go-v2/feature/ec2/imds v1.16.13/go.mod h1:NG7RXPUlqfsCLLFfi0+IpKN4sCB9D9fw/qTaSB+xRoU= +github.com/aws/aws-sdk-go-v2/feature/s3/manager v1.17.18 h1:9DIp7vhmOPmueCDwpXa45bEbLHHTt1kcxChdTJWWxvI= +github.com/aws/aws-sdk-go-v2/feature/s3/manager v1.17.18/go.mod h1:aJv/Fwz8r56ozwYFRC4bzoeL1L17GYQYemfblOBux1M= +github.com/aws/aws-sdk-go-v2/internal/configsources v1.3.17 h1:pI7Bzt0BJtYA0N/JEC6B8fJ4RBrEMi1LBrkMdFYNSnQ= +github.com/aws/aws-sdk-go-v2/internal/configsources v1.3.17/go.mod h1:Dh5zzJYMtxfIjYW+/evjQ8uj2OyR/ve2KROHGHlSFqE= +github.com/aws/aws-sdk-go-v2/internal/endpoints/v2 v2.6.17 h1:Mqr/V5gvrhA2gvgnF42Zh5iMiQNcOYthFYwCyrnuWlc= +github.com/aws/aws-sdk-go-v2/internal/endpoints/v2 v2.6.17/go.mod h1:aLJpZlCmjE+V+KtN1q1uyZkfnUWpQGpbsn89XPKyzfU= +github.com/aws/aws-sdk-go-v2/internal/ini v1.8.1 h1:VaRN3TlFdd6KxX1x3ILT5ynH6HvKgqdiXoTxAF4HQcQ= +github.com/aws/aws-sdk-go-v2/internal/ini v1.8.1/go.mod h1:FbtygfRFze9usAadmnGJNc8KsP346kEe+y2/oyhGAGc= +github.com/aws/aws-sdk-go-v2/internal/v4a v1.3.17 h1:Roo69qTpfu8OlJ2Tb7pAYVuF0CpuUMB0IYWwYP/4DZM= +github.com/aws/aws-sdk-go-v2/internal/v4a v1.3.17/go.mod h1:NcWPxQzGM1USQggaTVwz6VpqMZPX1CvDJLDh6jnOCa4= +github.com/aws/aws-sdk-go-v2/service/internal/accept-encoding v1.11.4 h1:KypMCbLPPHEmf9DgMGw51jMj77VfGPAN2Kv4cfhlfgI= +github.com/aws/aws-sdk-go-v2/service/internal/accept-encoding v1.11.4/go.mod h1:Vz1JQXliGcQktFTN/LN6uGppAIRoLBR2bMvIMP0gOjc= +github.com/aws/aws-sdk-go-v2/service/internal/checksum v1.3.19 h1:FLMkfEiRjhgeDTCjjLoc3URo/TBkgeQbocA78lfkzSI= +github.com/aws/aws-sdk-go-v2/service/internal/checksum v1.3.19/go.mod h1:Vx+GucNSsdhaxs3aZIKfSUjKVGsxN25nX2SRcdhuw08= +github.com/aws/aws-sdk-go-v2/service/internal/presigned-url v1.11.19 h1:rfprUlsdzgl7ZL2KlXiUAoJnI/VxfHCvDFr2QDFj6u4= +github.com/aws/aws-sdk-go-v2/service/internal/presigned-url v1.11.19/go.mod h1:SCWkEdRq8/7EK60NcvvQ6NXKuTcchAD4ROAsC37VEZE= +github.com/aws/aws-sdk-go-v2/service/internal/s3shared v1.17.17 h1:u+EfGmksnJc/x5tq3A+OD7LrMbSSR/5TrKLvkdy/fhY= +github.com/aws/aws-sdk-go-v2/service/internal/s3shared v1.17.17/go.mod h1:VaMx6302JHax2vHJWgRo+5n9zvbacs3bLU/23DNQrTY= +github.com/aws/aws-sdk-go-v2/service/s3 v1.61.2 h1:Kp6PWAlXwP1UvIflkIP6MFZYBNDCa4mFCGtxrpICVOg= +github.com/aws/aws-sdk-go-v2/service/s3 v1.61.2/go.mod h1:5FmD/Dqq57gP+XwaUnd5WFPipAuzrf0HmupX27Gvjvc= +github.com/aws/aws-sdk-go-v2/service/sso v1.22.7 h1:pIaGg+08llrP7Q5aiz9ICWbY8cqhTkyy+0SHvfzQpTc= +github.com/aws/aws-sdk-go-v2/service/sso v1.22.7/go.mod h1:eEygMHnTKH/3kNp9Jr1n3PdejuSNcgwLe1dWgQtO0VQ= +github.com/aws/aws-sdk-go-v2/service/ssooidc v1.26.7 h1:/Cfdu0XV3mONYKaOt1Gr0k1KvQzkzPyiKUdlWJqy+J4= +github.com/aws/aws-sdk-go-v2/service/ssooidc v1.26.7/go.mod h1:bCbAxKDqNvkHxRaIMnyVPXPo+OaPRwvmgzMxbz1VKSA= +github.com/aws/aws-sdk-go-v2/service/sts v1.30.7 h1:NKTa1eqZYw8tiHSRGpP0VtTdub/8KNk8sDkNPFaOKDE= +github.com/aws/aws-sdk-go-v2/service/sts v1.30.7/go.mod h1:NXi1dIAGteSaRLqYgarlhP/Ij0cFT+qmCwiJqWh/U5o= +github.com/aws/smithy-go v1.20.4 h1:2HK1zBdPgRbjFOHlfeQZfpC4r72MOb9bZkiFwggKO+4= +github.com/aws/smithy-go v1.20.4/go.mod h1:irrKGvNn1InZwb2d7fkIRNucdfwR8R+Ts3wxYa/cJHg= github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c= github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= github.com/essentialkaos/check v1.4.0 h1:kWdFxu9odCxUqo1NNFNJmguGrDHgwi3A8daXX1nkuKk= +github.com/essentialkaos/check v1.4.0/go.mod h1:LMKPZ2H+9PXe7Y2gEoKyVAwUqXVgx7KtgibfsHJPus0= github.com/essentialkaos/depsy v1.3.0 h1:CN7bRgBU2jGTHSkg/Sh38eDUn7cvmaTp2sxFt2HpFeU= github.com/essentialkaos/depsy v1.3.0/go.mod h1:kpiTAV17dyByVnrbNaMcZt2jRwvuXClUYOzpyJQwtG8= -github.com/essentialkaos/ek/v13 v13.2.0 h1:Ra6segoyFYjtdz5eh0mQxJMeIso7h61A7IyG9B4R6bI= -github.com/essentialkaos/ek/v13 v13.2.0/go.mod h1:RVf1NpNyK04xkBJ3NTUD1wNLWemY9/naVD4iEVjU2fA= +github.com/essentialkaos/ek/v13 v13.3.5 h1:k/lfDIpgE+8dOc+eF37m1hOi8eEMHYYTEzD5KHH3xxU= +github.com/essentialkaos/ek/v13 v13.3.5/go.mod h1:FWHXQ74YiSCsi/H9kmHR68fUE8aYUNcp0Fzz98XGUtM= github.com/essentialkaos/katana v0.2.0 h1:LRnKyEHFET9P45L718DI704oUBHcOjW+/bWBstPb9qg= github.com/essentialkaos/katana v0.2.0/go.mod h1:B0IUikFvR6Iutx93iSu3xezHfHvIuIgXJSO6Agujp+0= github.com/essentialkaos/sio v1.0.0 h1:+VZg0Z7+Cx8F/FmlczzTJYM6rq/LhTR45Rsditmu0Ec= @@ -37,12 +52,15 @@ github.com/essentialkaos/sio v1.0.0/go.mod h1:lKaW6IPMJ8GAEAiXe175zcEld370u3Nr54 github.com/kr/fs v0.1.0 h1:Jskdu9ieNAYnjxsi0LbQp1ulIKZV1LAFgK1tWhpZgl8= github.com/kr/fs v0.1.0/go.mod h1:FFnZGqtBN9Gxj7eW1uZ42v5BccTP0vu6NEaFoC2HwRg= github.com/kr/pretty v0.3.1 h1:flRD4NNwYAUpkphVc1HcthR4KEIFJ65n8Mw5qdRn3LE= +github.com/kr/pretty v0.3.1/go.mod h1:hoEshYVHaxMs3cyo3Yncou5ZscifuDolrwPKZanG3xk= github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY= +github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE= github.com/pkg/sftp v1.13.6 h1:JFZT4XbOU7l77xGSpOdW+pwIMqP044IyjXX6FGyEKFo= github.com/pkg/sftp v1.13.6/go.mod h1:tz1ryNURKu77RL+GuCzmoJYxQczL3wLNNpPWagdg4Qk= github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= github.com/rogpeppe/go-internal v1.11.0 h1:cWPaGQEPrBb5/AsnsZesgZZ9yb1OQ+GOISoDNXVBh4M= +github.com/rogpeppe/go-internal v1.11.0/go.mod h1:ddIwULY96R17DhadqLgMfk9H9tvdUzkipdSkR5nkCZA= github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= github.com/stretchr/objx v0.4.0/go.mod h1:YvHI0jy2hoMjB+UWwv71VJQ9isScKT/TqJzVSSt89Yw= github.com/stretchr/testify v1.7.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= @@ -52,8 +70,8 @@ github.com/yuin/goldmark v1.4.13/go.mod h1:6yULJ656Px+3vBD8DxQVa3kxgyrAnzto9xy5t golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w= golang.org/x/crypto v0.0.0-20210921155107-089bfa567519/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc= golang.org/x/crypto v0.1.0/go.mod h1:RecgLatLF4+eUMCP1PoPZQb+cVrJcOPbHkTkbkB9sbw= -golang.org/x/crypto v0.25.0 h1:ypSNr+bnYL2YhwoMt2zPxHFmbAN1KZs/njMG3hxUp30= -golang.org/x/crypto v0.25.0/go.mod h1:T+wALwcMOSE0kXgUAnPAHqTLW+XHgcELELW8VaDgm/M= +golang.org/x/crypto v0.26.0 h1:RrRspgV4mU+YwB4FYnuBoKsUapNIL5cohGAmSH3azsw= +golang.org/x/crypto v0.26.0/go.mod h1:GY7jblb9wI+FOo5y8/S2oY4zWP07AkOJ4+jxCqdqn54= golang.org/x/mod v0.6.0-dev.0.20220419223038-86c51ed26bb4/go.mod h1:jJ57K6gSWd91VN4djpZkiMVwK6gcyfeH4XE8wZrZaV4= golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= golang.org/x/net v0.0.0-20210226172049-e18ecbb05110/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= @@ -67,12 +85,13 @@ golang.org/x/sys v0.0.0-20210615035016-665e8c7367d1/go.mod h1:oPkhp1MJrh7nUepCBc golang.org/x/sys v0.0.0-20220520151302-bc2c85ada10a/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20220722155257-8c9f86f7a55f/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.1.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/sys v0.22.0 h1:RI27ohtqKCnwULzJLqkv897zojh5/DwS/ENaMzUOaWI= -golang.org/x/sys v0.22.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA= +golang.org/x/sys v0.24.0 h1:Twjiwq9dn6R1fQcyiK+wQyHWfaz/BJB+YIpzU/Cv3Xg= +golang.org/x/sys v0.24.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA= golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= golang.org/x/term v0.0.0-20210927222741-03fcf44c2211/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8= golang.org/x/term v0.1.0/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8= -golang.org/x/term v0.22.0 h1:BbsgPEJULsl2fV/AT3v15Mjva5yXKQDyKf+TbDz7QJk= +golang.org/x/term v0.23.0 h1:F6D4vR+EHoL9/sWAWgAR1H2DcHr4PareCbAaCo1RpuU= +golang.org/x/term v0.23.0/go.mod h1:DgV24QBUrK6jhZXl+20l6UWznPlwAHm1Q1mGHtydmSk= golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= golang.org/x/text v0.3.7/go.mod h1:u+2+/6zg+i71rQMx5EYifcz6MCKuco9NR6JIITiCfzQ= diff --git a/uploader/fs/fs.go b/uploader/fs/fs.go index f9166b2..9f79226 100644 --- a/uploader/fs/fs.go +++ b/uploader/fs/fs.go @@ -29,9 +29,9 @@ import ( // Config is configuration for FS uploader type Config struct { + Secret *katana.Secret Path string Mode os.FileMode - Secret *katana.Secret } // FSUploader is FS uploader instance diff --git a/uploader/s3/s3.go b/uploader/s3/s3.go index 669682b..e188497 100644 --- a/uploader/s3/s3.go +++ b/uploader/s3/s3.go @@ -23,6 +23,7 @@ import ( "github.com/aws/aws-sdk-go-v2/aws" "github.com/aws/aws-sdk-go-v2/credentials" + "github.com/aws/aws-sdk-go-v2/feature/s3/manager" "github.com/aws/aws-sdk-go-v2/service/s3" "github.com/essentialkaos/katana" @@ -34,13 +35,15 @@ import ( // Config is configuration for S3 uploader type Config struct { + Secret *katana.Secret + Host string Region string AccessKeyID string SecretKey string Bucket string Path string - Secret *katana.Secret + PartSize int64 } // S3Uploader is S3 uploader instance @@ -155,7 +158,11 @@ func (u *S3Uploader) Write(r io.ReadCloser, fileName string, fileSize int64) err )), }) - _, err = client.PutObject(context.TODO(), &s3.PutObjectInput{ + manager := manager.NewUploader(client, func(c *manager.Uploader) { + c.PartSize = u.config.PartSize * 1024 * 1024 + }) + + _, err = manager.Upload(context.TODO(), &s3.PutObjectInput{ Bucket: aws.String(u.config.Bucket), Key: aws.String(outputFile), Body: rr, diff --git a/uploader/sftp/sftp.go b/uploader/sftp/sftp.go index 08e7ee1..11c1148 100644 --- a/uploader/sftp/sftp.go +++ b/uploader/sftp/sftp.go @@ -32,12 +32,12 @@ import ( // Config is configuration for SFTP uploader type Config struct { + Secret *katana.Secret Host string User string Key []byte Path string Mode os.FileMode - Secret *katana.Secret } // SFTPUploader is SFTP uploader instance