Compare commits
3 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
7ceb4809a0 | ||
|
|
6ab520723c | ||
|
|
18cdce0df8 |
@ -39,6 +39,7 @@ var serviceCommands = []func(cmd *cobra.Command) *cobra.Command{
|
|||||||
addExchangeCommands,
|
addExchangeCommands,
|
||||||
addOneDriveCommands,
|
addOneDriveCommands,
|
||||||
addSharePointCommands,
|
addSharePointCommands,
|
||||||
|
addTeamsCommands,
|
||||||
}
|
}
|
||||||
|
|
||||||
// AddCommands attaches all `corso backup * *` commands to the parent.
|
// AddCommands attaches all `corso backup * *` commands to the parent.
|
||||||
|
|||||||
186
src/cli/backup/teams.go
Normal file
186
src/cli/backup/teams.go
Normal file
@ -0,0 +1,186 @@
|
|||||||
|
package backup
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/spf13/cobra"
|
||||||
|
"github.com/spf13/pflag"
|
||||||
|
|
||||||
|
"github.com/alcionai/corso/src/cli/flags"
|
||||||
|
"github.com/alcionai/corso/src/cli/utils"
|
||||||
|
"github.com/alcionai/corso/src/pkg/path"
|
||||||
|
)
|
||||||
|
|
||||||
|
// ------------------------------------------------------------------------------------------------
|
||||||
|
// setup and globals
|
||||||
|
// ------------------------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
const (
|
||||||
|
teamsServiceCommand = "teams"
|
||||||
|
teamsServiceCommandCreateUseSuffix = "--team <teamsName> | '" + flags.Wildcard + "'"
|
||||||
|
teamsServiceCommandDeleteUseSuffix = "--backup <backupId>"
|
||||||
|
teamsServiceCommandDetailsUseSuffix = "--backup <backupId>"
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
teamsServiceCommandCreateExamples = `# Backup all Teams data for Alice
|
||||||
|
corso backup create teams --team alice@example.com
|
||||||
|
|
||||||
|
# Backup only Teams contacts for Alice and Bob
|
||||||
|
corso backup create teams --team engineering,sales --data contacts
|
||||||
|
|
||||||
|
# Backup all Teams data for all M365 users
|
||||||
|
corso backup create teams --team '*'`
|
||||||
|
|
||||||
|
teamsServiceCommandDeleteExamples = `# Delete Teams backup with ID 1234abcd-12ab-cd34-56de-1234abcd
|
||||||
|
corso backup delete teams --backup 1234abcd-12ab-cd34-56de-1234abcd`
|
||||||
|
|
||||||
|
teamsServiceCommandDetailsExamples = `# Explore items in Alice's latest backup (1234abcd...)
|
||||||
|
corso backup details teams --backup 1234abcd-12ab-cd34-56de-1234abcd
|
||||||
|
|
||||||
|
# Explore calendar events occurring after start of 2022
|
||||||
|
corso backup details teams --backup 1234abcd-12ab-cd34-56de-1234abcd \
|
||||||
|
--event-starts-after 2022-01-01T00:00:00`
|
||||||
|
)
|
||||||
|
|
||||||
|
// called by backup.go to map subcommands to provider-specific handling.
|
||||||
|
func addTeamsCommands(cmd *cobra.Command) *cobra.Command {
|
||||||
|
var (
|
||||||
|
c *cobra.Command
|
||||||
|
fs *pflag.FlagSet
|
||||||
|
)
|
||||||
|
|
||||||
|
switch cmd.Use {
|
||||||
|
case createCommand:
|
||||||
|
c, fs = utils.AddCommand(cmd, teamsCreateCmd())
|
||||||
|
fs.SortFlags = false
|
||||||
|
|
||||||
|
c.Use = c.Use + " " + teamsServiceCommandCreateUseSuffix
|
||||||
|
c.Example = teamsServiceCommandCreateExamples
|
||||||
|
|
||||||
|
// Flags addition ordering should follow the order we want them to appear in help and docs:
|
||||||
|
// TODO Neha: add teams flag
|
||||||
|
flags.AddDataFlag(c, []string{dataEmail, dataContacts, dataEvents}, false)
|
||||||
|
flags.AddCorsoPassphaseFlags(c)
|
||||||
|
flags.AddAWSCredsFlags(c)
|
||||||
|
flags.AddAzureCredsFlags(c)
|
||||||
|
flags.AddFetchParallelismFlag(c)
|
||||||
|
flags.AddFailFastFlag(c)
|
||||||
|
|
||||||
|
case listCommand:
|
||||||
|
c, fs = utils.AddCommand(cmd, teamsListCmd())
|
||||||
|
fs.SortFlags = false
|
||||||
|
|
||||||
|
flags.AddBackupIDFlag(c, false)
|
||||||
|
flags.AddCorsoPassphaseFlags(c)
|
||||||
|
flags.AddAWSCredsFlags(c)
|
||||||
|
flags.AddAzureCredsFlags(c)
|
||||||
|
addFailedItemsFN(c)
|
||||||
|
addSkippedItemsFN(c)
|
||||||
|
addRecoveredErrorsFN(c)
|
||||||
|
|
||||||
|
case detailsCommand:
|
||||||
|
c, fs = utils.AddCommand(cmd, teamsDetailsCmd())
|
||||||
|
fs.SortFlags = false
|
||||||
|
|
||||||
|
c.Use = c.Use + " " + teamsServiceCommandDetailsUseSuffix
|
||||||
|
c.Example = teamsServiceCommandDetailsExamples
|
||||||
|
|
||||||
|
flags.AddSkipReduceFlag(c)
|
||||||
|
|
||||||
|
// Flags addition ordering should follow the order we want them to appear in help and docs:
|
||||||
|
// More generic (ex: --user) and more frequently used flags take precedence.
|
||||||
|
flags.AddBackupIDFlag(c, true)
|
||||||
|
flags.AddCorsoPassphaseFlags(c)
|
||||||
|
flags.AddAWSCredsFlags(c)
|
||||||
|
flags.AddAzureCredsFlags(c)
|
||||||
|
|
||||||
|
case deleteCommand:
|
||||||
|
c, fs = utils.AddCommand(cmd, teamsDeleteCmd())
|
||||||
|
fs.SortFlags = false
|
||||||
|
|
||||||
|
c.Use = c.Use + " " + teamsServiceCommandDeleteUseSuffix
|
||||||
|
c.Example = teamsServiceCommandDeleteExamples
|
||||||
|
|
||||||
|
flags.AddBackupIDFlag(c, true)
|
||||||
|
flags.AddCorsoPassphaseFlags(c)
|
||||||
|
flags.AddAWSCredsFlags(c)
|
||||||
|
flags.AddAzureCredsFlags(c)
|
||||||
|
}
|
||||||
|
|
||||||
|
return c
|
||||||
|
}
|
||||||
|
|
||||||
|
// ------------------------------------------------------------------------------------------------
|
||||||
|
// backup create
|
||||||
|
// ------------------------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
// `corso backup create teams [<flag>...]`
|
||||||
|
func teamsCreateCmd() *cobra.Command {
|
||||||
|
return &cobra.Command{
|
||||||
|
Use: teamsServiceCommand,
|
||||||
|
Short: "Backup M365 Team service data",
|
||||||
|
RunE: createTeamsCmd,
|
||||||
|
Args: cobra.NoArgs,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// processes a teams service backup.
|
||||||
|
func createTeamsCmd(cmd *cobra.Command, args []string) error {
|
||||||
|
return cmd.Help()
|
||||||
|
}
|
||||||
|
|
||||||
|
// ------------------------------------------------------------------------------------------------
|
||||||
|
// backup list
|
||||||
|
// ------------------------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
// `corso backup list teams [<flag>...]`
|
||||||
|
func teamsListCmd() *cobra.Command {
|
||||||
|
return &cobra.Command{
|
||||||
|
Use: teamsServiceCommand,
|
||||||
|
Short: "List the history of M365 Teams service backups",
|
||||||
|
RunE: listTeamsCmd,
|
||||||
|
Args: cobra.NoArgs,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// lists the history of backup operations
|
||||||
|
func listTeamsCmd(cmd *cobra.Command, args []string) error {
|
||||||
|
return genericListCommand(cmd, flags.BackupIDFV, path.TeamsService, args)
|
||||||
|
}
|
||||||
|
|
||||||
|
// ------------------------------------------------------------------------------------------------
|
||||||
|
// backup details
|
||||||
|
// ------------------------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
// `corso backup details teams [<flag>...]`
|
||||||
|
func teamsDetailsCmd() *cobra.Command {
|
||||||
|
return &cobra.Command{
|
||||||
|
Use: teamsServiceCommand,
|
||||||
|
Short: "Shows the details of a M365 Teams service backup",
|
||||||
|
RunE: detailsTeamsCmd,
|
||||||
|
Args: cobra.NoArgs,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// processes a teams service backup.
|
||||||
|
func detailsTeamsCmd(cmd *cobra.Command, args []string) error {
|
||||||
|
return cmd.Help()
|
||||||
|
}
|
||||||
|
|
||||||
|
// ------------------------------------------------------------------------------------------------
|
||||||
|
// backup delete
|
||||||
|
// ------------------------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
// `corso backup delete teams [<flag>...]`
|
||||||
|
func teamsDeleteCmd() *cobra.Command {
|
||||||
|
return &cobra.Command{
|
||||||
|
Use: teamsServiceCommand,
|
||||||
|
Short: "Delete backed-up M365 Teams service data",
|
||||||
|
RunE: deleteTeamsCmd,
|
||||||
|
Args: cobra.NoArgs,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// deletes an teams service backup.
|
||||||
|
func deleteTeamsCmd(cmd *cobra.Command, args []string) error {
|
||||||
|
return genericDeleteCommand(cmd, path.TeamsService, flags.BackupIDFV, "Teams", args)
|
||||||
|
}
|
||||||
98
src/cli/backup/teams_test.go
Normal file
98
src/cli/backup/teams_test.go
Normal file
@ -0,0 +1,98 @@
|
|||||||
|
package backup
|
||||||
|
|
||||||
|
import (
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/spf13/cobra"
|
||||||
|
"github.com/stretchr/testify/assert"
|
||||||
|
"github.com/stretchr/testify/require"
|
||||||
|
"github.com/stretchr/testify/suite"
|
||||||
|
|
||||||
|
"github.com/alcionai/corso/src/cli/flags"
|
||||||
|
"github.com/alcionai/corso/src/internal/tester"
|
||||||
|
)
|
||||||
|
|
||||||
|
type TeamsUnitSuite struct {
|
||||||
|
tester.Suite
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestTeamsUnitSuite(t *testing.T) {
|
||||||
|
suite.Run(t, &TeamsUnitSuite{Suite: tester.NewUnitSuite(t)})
|
||||||
|
}
|
||||||
|
|
||||||
|
func (suite *TeamsUnitSuite) TestAddTeamsCommands() {
|
||||||
|
expectUse := teamsServiceCommand
|
||||||
|
|
||||||
|
table := []struct {
|
||||||
|
name string
|
||||||
|
use string
|
||||||
|
expectUse string
|
||||||
|
expectShort string
|
||||||
|
flags []string
|
||||||
|
expectRunE func(*cobra.Command, []string) error
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
"create teams",
|
||||||
|
createCommand,
|
||||||
|
expectUse + " " + teamsServiceCommandCreateUseSuffix,
|
||||||
|
teamsCreateCmd().Short,
|
||||||
|
[]string{
|
||||||
|
flags.CategoryDataFN,
|
||||||
|
flags.FailFastFN,
|
||||||
|
flags.FetchParallelismFN,
|
||||||
|
flags.SkipReduceFN,
|
||||||
|
flags.NoStatsFN,
|
||||||
|
},
|
||||||
|
createTeamsCmd,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"list teams",
|
||||||
|
listCommand,
|
||||||
|
expectUse,
|
||||||
|
teamsListCmd().Short,
|
||||||
|
[]string{
|
||||||
|
flags.BackupFN,
|
||||||
|
flags.FailedItemsFN,
|
||||||
|
flags.SkippedItemsFN,
|
||||||
|
flags.RecoveredErrorsFN,
|
||||||
|
},
|
||||||
|
listTeamsCmd,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"details teams",
|
||||||
|
detailsCommand,
|
||||||
|
expectUse + " " + teamsServiceCommandDetailsUseSuffix,
|
||||||
|
teamsDetailsCmd().Short,
|
||||||
|
[]string{
|
||||||
|
flags.BackupFN,
|
||||||
|
},
|
||||||
|
detailsTeamsCmd,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"delete teams",
|
||||||
|
deleteCommand,
|
||||||
|
expectUse + " " + teamsServiceCommandDeleteUseSuffix,
|
||||||
|
teamsDeleteCmd().Short,
|
||||||
|
[]string{flags.BackupFN},
|
||||||
|
deleteTeamsCmd,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
for _, test := range table {
|
||||||
|
suite.Run(test.name, func() {
|
||||||
|
t := suite.T()
|
||||||
|
|
||||||
|
cmd := &cobra.Command{Use: test.use}
|
||||||
|
|
||||||
|
c := addTeamsCommands(cmd)
|
||||||
|
require.NotNil(t, c)
|
||||||
|
|
||||||
|
cmds := cmd.Commands()
|
||||||
|
require.Len(t, cmds, 1)
|
||||||
|
|
||||||
|
child := cmds[0]
|
||||||
|
assert.Equal(t, test.expectUse, child.Use)
|
||||||
|
assert.Equal(t, test.expectShort, child.Short)
|
||||||
|
tester.AreSameFunc(t, test.expectRunE, child.RunE)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -1,4 +1,4 @@
|
|||||||
// Code generated by "stringer -type=opStatus -linecomment"; DO NOT EDIT.
|
// Code generated by "stringer -type=OpStatus -linecomment"; DO NOT EDIT.
|
||||||
|
|
||||||
package operations
|
package operations
|
||||||
|
|
||||||
@ -15,13 +15,13 @@ func _() {
|
|||||||
_ = x[NoData-4]
|
_ = x[NoData-4]
|
||||||
}
|
}
|
||||||
|
|
||||||
const _opStatus_name = "Status UnknownIn ProgressCompletedFailedNo Data"
|
const _OpStatus_name = "Status UnknownIn ProgressCompletedFailedNo Data"
|
||||||
|
|
||||||
var _opStatus_index = [...]uint8{0, 14, 25, 34, 40, 47}
|
var _OpStatus_index = [...]uint8{0, 14, 25, 34, 40, 47}
|
||||||
|
|
||||||
func (i OpStatus) String() string {
|
func (i OpStatus) String() string {
|
||||||
if i < 0 || i >= OpStatus(len(_opStatus_index)-1) {
|
if i < 0 || i >= OpStatus(len(_OpStatus_index)-1) {
|
||||||
return "opStatus(" + strconv.FormatInt(int64(i), 10) + ")"
|
return "OpStatus(" + strconv.FormatInt(int64(i), 10) + ")"
|
||||||
}
|
}
|
||||||
return _opStatus_name[_opStatus_index[i]:_opStatus_index[i+1]]
|
return _OpStatus_name[_OpStatus_index[i]:_OpStatus_index[i+1]]
|
||||||
}
|
}
|
||||||
|
|||||||
@ -30,6 +30,8 @@ const (
|
|||||||
ExchangeMetadataService // exchangeMetadata
|
ExchangeMetadataService // exchangeMetadata
|
||||||
OneDriveMetadataService // onedriveMetadata
|
OneDriveMetadataService // onedriveMetadata
|
||||||
SharePointMetadataService // sharepointMetadata
|
SharePointMetadataService // sharepointMetadata
|
||||||
|
TeamsService // teams
|
||||||
|
TeamsMetadataService // teamsMetadata
|
||||||
)
|
)
|
||||||
|
|
||||||
func toServiceType(service string) ServiceType {
|
func toServiceType(service string) ServiceType {
|
||||||
|
|||||||
@ -15,11 +15,13 @@ func _() {
|
|||||||
_ = x[ExchangeMetadataService-4]
|
_ = x[ExchangeMetadataService-4]
|
||||||
_ = x[OneDriveMetadataService-5]
|
_ = x[OneDriveMetadataService-5]
|
||||||
_ = x[SharePointMetadataService-6]
|
_ = x[SharePointMetadataService-6]
|
||||||
|
_ = x[TeamsService-7]
|
||||||
|
_ = x[TeamsMetadataService-8]
|
||||||
}
|
}
|
||||||
|
|
||||||
const _ServiceType_name = "UnknownServiceexchangeonedrivesharepointexchangeMetadataonedriveMetadatasharepointMetadata"
|
const _ServiceType_name = "UnknownServiceexchangeonedrivesharepointexchangeMetadataonedriveMetadatasharepointMetadatateamsteamsMetadata"
|
||||||
|
|
||||||
var _ServiceType_index = [...]uint8{0, 14, 22, 30, 40, 56, 72, 90}
|
var _ServiceType_index = [...]uint8{0, 14, 22, 30, 40, 56, 72, 90, 95, 108}
|
||||||
|
|
||||||
func (i ServiceType) String() string {
|
func (i ServiceType) String() string {
|
||||||
if i < 0 || i >= ServiceType(len(_ServiceType_index)-1) {
|
if i < 0 || i >= ServiceType(len(_ServiceType_index)-1) {
|
||||||
|
|||||||
Loading…
x
Reference in New Issue
Block a user