Sanity test cleanups and additional info (#2816)
updating sanity tests: fix the permissions check by not using reflect.DeepEquals, and various code cleanup. --- #### Does this PR need a docs update or release note? - [x] ⛔ No #### Type of change - [x] 🤖 Test #### Test Plan - [x] 💪 Manual - [x] 💚 E2E
This commit is contained in:
parent
47067bf7ab
commit
ddeaaf2ff2
2
.github/workflows/load_test.yml
vendored
2
.github/workflows/load_test.yml
vendored
@ -137,7 +137,7 @@ jobs:
|
|||||||
strategy:
|
strategy:
|
||||||
matrix:
|
matrix:
|
||||||
user: ${{ fromJson(needs.setup.outputs.matrix).user }}
|
user: ${{ fromJson(needs.setup.outputs.matrix).user }}
|
||||||
folder: [Corso_Restore_,'']
|
folder: [Corso_Restore_, '']
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v3
|
- uses: actions/checkout@v3
|
||||||
- name: Set folder boundary datetime
|
- name: Set folder boundary datetime
|
||||||
|
|||||||
138
.github/workflows/sanity-test.yaml
vendored
138
.github/workflows/sanity-test.yaml
vendored
@ -18,21 +18,21 @@ concurrency:
|
|||||||
group: sanity_testing-${{ github.workflow }}-${{ github.ref }}
|
group: sanity_testing-${{ github.workflow }}-${{ github.ref }}
|
||||||
cancel-in-progress: true
|
cancel-in-progress: true
|
||||||
|
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
Sanity-Tests:
|
Sanity-Tests:
|
||||||
environment: Testing
|
environment: Testing
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
env:
|
env:
|
||||||
|
AWS_ACCESS_KEY_ID: ${{ secrets.AWS_ACCESS_KEY_ID }}
|
||||||
|
AWS_SECRET_ACCESS_KEY: ${{ secrets.AWS_ACCESS_KEY_SECRET }}
|
||||||
AZURE_CLIENT_ID: ${{ secrets.CLIENT_ID }}
|
AZURE_CLIENT_ID: ${{ secrets.CLIENT_ID }}
|
||||||
AZURE_CLIENT_SECRET: ${{ secrets.CLIENT_SECRET }}
|
AZURE_CLIENT_SECRET: ${{ secrets.CLIENT_SECRET }}
|
||||||
AZURE_TENANT_ID: ${{ secrets.TENANT_ID }}
|
AZURE_TENANT_ID: ${{ secrets.TENANT_ID }}
|
||||||
|
CORSO_BUCKET: ${{ secrets.CI_TESTS_S3_BUCKET }}
|
||||||
|
CORSO_LOG_FILE: ./src/testlog/testlogging.log
|
||||||
CORSO_M365_TEST_USER_ID: ${{ github.event.inputs.user != '' && github.event.inputs.user || secrets.CORSO_M365_TEST_USER_ID }}
|
CORSO_M365_TEST_USER_ID: ${{ github.event.inputs.user != '' && github.event.inputs.user || secrets.CORSO_M365_TEST_USER_ID }}
|
||||||
CORSO_PASSPHRASE: ${{ secrets.INTEGRATION_TEST_CORSO_PASSPHRASE }}
|
CORSO_PASSPHRASE: ${{ secrets.INTEGRATION_TEST_CORSO_PASSPHRASE }}
|
||||||
TEST_RESULT: "test_results"
|
TEST_RESULT: "test_results"
|
||||||
CORSO_BUCKET: ${{ secrets.CI_TESTS_S3_BUCKET }}
|
|
||||||
AWS_ACCESS_KEY_ID: ${{ secrets.AWS_ACCESS_KEY_ID }}
|
|
||||||
AWS_SECRET_ACCESS_KEY: ${{ secrets.AWS_ACCESS_KEY_SECRET }}
|
|
||||||
|
|
||||||
defaults:
|
defaults:
|
||||||
run:
|
run:
|
||||||
@ -76,8 +76,8 @@ jobs:
|
|||||||
|
|
||||||
if ! grep -q 'Initialized a S3 repository within bucket' $TEST_RESULT/initrepo.txt
|
if ! grep -q 'Initialized a S3 repository within bucket' $TEST_RESULT/initrepo.txt
|
||||||
then
|
then
|
||||||
echo "repo could not be initiated"
|
echo "repo could not be initiated"
|
||||||
exit 1
|
exit 1
|
||||||
fi
|
fi
|
||||||
|
|
||||||
echo result="$prefix" >> $GITHUB_OUTPUT
|
echo result="$prefix" >> $GITHUB_OUTPUT
|
||||||
@ -93,8 +93,8 @@ jobs:
|
|||||||
|
|
||||||
if ! grep -q 'Connected to S3 bucket' $TEST_RESULT/connect.txt
|
if ! grep -q 'Connected to S3 bucket' $TEST_RESULT/connect.txt
|
||||||
then
|
then
|
||||||
echo "repo could not be connected"
|
echo "repo could not be connected"
|
||||||
exit 1
|
exit 1
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# run the tests
|
# run the tests
|
||||||
@ -103,29 +103,47 @@ jobs:
|
|||||||
run: |
|
run: |
|
||||||
./corso backup create exchange \
|
./corso backup create exchange \
|
||||||
--user "${CORSO_M365_TEST_USER_ID}" \
|
--user "${CORSO_M365_TEST_USER_ID}" \
|
||||||
--hide-progress --json 2>&1 | tee $TEST_RESULT/backup_exchange.txt
|
--hide-progress \
|
||||||
|
--json \
|
||||||
|
2>&1 | tee $TEST_RESULT/backup_exchange.txt
|
||||||
|
|
||||||
resultjson=$(sed -e '1,/Completed Backups/d' $TEST_RESULT/backup_exchange.txt )
|
resultjson=$(sed -e '1,/Completed Backups/d' $TEST_RESULT/backup_exchange.txt )
|
||||||
|
|
||||||
if [[ $( echo $resultjson | jq -r '.[0] | .errorCount') -ne 0 ]]; then
|
if [[ $( echo $resultjson | jq -r '.[0] | .errorCount') -ne 0 ]]; then
|
||||||
echo "backup was not successfull"
|
echo "backup was not successful"
|
||||||
exit 1
|
exit 1
|
||||||
fi
|
fi
|
||||||
|
|
||||||
data=$( echo $resultjson | jq -r '.[0] | .id' )
|
data=$( echo $resultjson | jq -r '.[0] | .id' )
|
||||||
echo result=$data >> $GITHUB_OUTPUT
|
echo result=$data >> $GITHUB_OUTPUT
|
||||||
|
|
||||||
# list the backup exhange
|
# list all exchange backups
|
||||||
- name: Backup exchange list test
|
- name: Backup exchange list test
|
||||||
run: |
|
run: |
|
||||||
set -euo pipefail
|
set -euo pipefail
|
||||||
./corso backup list exchange \
|
./corso backup list exchange \
|
||||||
--hide-progress 2>&1 | tee $TEST_RESULT/backup_exchange_list.txt
|
--hide-progress \
|
||||||
|
2>&1 | tee $TEST_RESULT/backup_exchange_list.txt
|
||||||
|
|
||||||
if ! grep -q ${{ steps.exchange-test.outputs.result }} $TEST_RESULT/backup_exchange_list.txt
|
if ! grep -q ${{ steps.exchange-test.outputs.result }} $TEST_RESULT/backup_exchange_list.txt
|
||||||
then
|
then
|
||||||
echo "listing of backup was not successfull"
|
echo "listing of backup was not successful"
|
||||||
exit 1
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
# list the previous exchange backups
|
||||||
|
- name: Backup exchange list single backup test
|
||||||
|
run: |
|
||||||
|
set -euo pipefail
|
||||||
|
./corso backup list exchange \
|
||||||
|
--hide-progress \
|
||||||
|
--backup "${{ steps.exchange-test.outputs.result }}" \
|
||||||
|
2>&1 | tee $TEST_RESULT/backup_exchange_list_single.txt
|
||||||
|
|
||||||
|
if ! grep -q ${{ steps.exchange-test.outputs.result }} $TEST_RESULT/backup_exchange_list.txt
|
||||||
|
then
|
||||||
|
echo "listing of backup was not successful"
|
||||||
|
exit 1
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# test exchange restore
|
# test exchange restore
|
||||||
@ -135,31 +153,34 @@ jobs:
|
|||||||
set -euo pipefail
|
set -euo pipefail
|
||||||
./corso restore exchange \
|
./corso restore exchange \
|
||||||
--hide-progress \
|
--hide-progress \
|
||||||
--backup "${{ steps.exchange-test.outputs.result }}" 2>&1 | tee $TEST_RESULT/exchange-restore-test.txt
|
--backup "${{ steps.exchange-test.outputs.result }}" \
|
||||||
|
2>&1 | tee $TEST_RESULT/exchange-restore-test.txt
|
||||||
echo result=$(grep -i -e 'Restoring to folder ' $TEST_RESULT/exchange-restore-test.txt | sed "s/Restoring to folder//" ) >> $GITHUB_OUTPUT
|
echo result=$(grep -i -e 'Restoring to folder ' $TEST_RESULT/exchange-restore-test.txt | sed "s/Restoring to folder//" ) >> $GITHUB_OUTPUT
|
||||||
|
|
||||||
- name: Restoration check
|
- name: Restoration check
|
||||||
env:
|
env:
|
||||||
RESTORE_FOLDER: ${{ steps.exchange-restore-test.outputs.result }}
|
SANITY_RESTORE_FOLDER: ${{ steps.exchange-restore-test.outputs.result }}
|
||||||
RESTORE_SERVICE: "exchange"
|
SANITY_RESTORE_SERVICE: "exchange"
|
||||||
run: |
|
run: |
|
||||||
set -euo pipefail
|
set -euo pipefail
|
||||||
./sanityCheck
|
./sanityCheck
|
||||||
|
|
||||||
# test incremental backup exhange
|
# test incremental backup exchange
|
||||||
- name: Backup exchange incremental
|
- name: Backup exchange incremental
|
||||||
id: exchange-incremental-test
|
id: exchange-incremental-test
|
||||||
run: |
|
run: |
|
||||||
set -euo pipefail
|
set -euo pipefail
|
||||||
./corso backup create exchange \
|
./corso backup create exchange \
|
||||||
|
--hide-progress \
|
||||||
--user "${CORSO_M365_TEST_USER_ID}" \
|
--user "${CORSO_M365_TEST_USER_ID}" \
|
||||||
--hide-progress --json 2>&1 | tee $TEST_RESULT/backup_exchange_incremental.txt
|
--json \
|
||||||
|
2>&1 | tee $TEST_RESULT/backup_exchange_incremental.txt
|
||||||
|
|
||||||
resultjson=$(sed -e '1,/Completed Backups/d' $TEST_RESULT/backup_exchange_incremental.txt )
|
resultjson=$(sed -e '1,/Completed Backups/d' $TEST_RESULT/backup_exchange_incremental.txt )
|
||||||
|
|
||||||
if [[ $( echo $resultjson | jq -r '.[0] | .errorCount') -ne 0 ]]; then
|
if [[ $( echo $resultjson | jq -r '.[0] | .errorCount') -ne 0 ]]; then
|
||||||
echo "backup was not successfull"
|
echo "backup was not successful"
|
||||||
exit 1
|
exit 1
|
||||||
fi
|
fi
|
||||||
|
|
||||||
echo result=$( echo $resultjson | jq -r '.[0] | .id' ) >> $GITHUB_OUTPUT
|
echo result=$( echo $resultjson | jq -r '.[0] | .id' ) >> $GITHUB_OUTPUT
|
||||||
@ -171,13 +192,14 @@ jobs:
|
|||||||
set -euo pipefail
|
set -euo pipefail
|
||||||
./corso restore exchange \
|
./corso restore exchange \
|
||||||
--hide-progress \
|
--hide-progress \
|
||||||
--backup "${{ steps.exchange-incremental-test.outputs.result }}" 2>&1 | tee $TEST_RESULT/exchange-incremantal-restore-test.txt
|
--backup "${{ steps.exchange-incremental-test.outputs.result }}" \
|
||||||
|
2>&1 | tee $TEST_RESULT/exchange-incremantal-restore-test.txt
|
||||||
echo result=$(grep -i -e 'Restoring to folder ' $TEST_RESULT/exchange-incremantal-restore-test.txt | sed "s/Restoring to folder//" ) >> $GITHUB_OUTPUT
|
echo result=$(grep -i -e 'Restoring to folder ' $TEST_RESULT/exchange-incremantal-restore-test.txt | sed "s/Restoring to folder//" ) >> $GITHUB_OUTPUT
|
||||||
|
|
||||||
- name: Restoration check
|
- name: Restoration check
|
||||||
env:
|
env:
|
||||||
RESTORE_FOLDER: ${{ steps.exchange-incremantal-restore-test.outputs.result }}
|
SANITY_RESTORE_FOLDER: ${{ steps.exchange-incremantal-restore-test.outputs.result }}
|
||||||
RESTORE_SERVICE: "exchange"
|
SANITY_RESTORE_SERVICE: "exchange"
|
||||||
run: |
|
run: |
|
||||||
set -euo pipefail
|
set -euo pipefail
|
||||||
./sanityCheck
|
./sanityCheck
|
||||||
@ -191,30 +213,48 @@ jobs:
|
|||||||
run: |
|
run: |
|
||||||
set -euo pipefail
|
set -euo pipefail
|
||||||
./corso backup create onedrive \
|
./corso backup create onedrive \
|
||||||
|
--hide-progress \
|
||||||
--user "${CORSO_M365_TEST_USER_ID}" \
|
--user "${CORSO_M365_TEST_USER_ID}" \
|
||||||
--hide-progress --json 2>&1 | tee $TEST_RESULT/backup_onedrive.txt
|
--json \
|
||||||
|
2>&1 | tee $TEST_RESULT/backup_onedrive.txt
|
||||||
|
|
||||||
resultjson=$(sed -e '1,/Completed Backups/d' $TEST_RESULT/backup_onedrive.txt )
|
resultjson=$(sed -e '1,/Completed Backups/d' $TEST_RESULT/backup_onedrive.txt )
|
||||||
|
|
||||||
if [[ $( echo $resultjson | jq -r '.[0] | .errorCount') -ne 0 ]]; then
|
if [[ $( echo $resultjson | jq -r '.[0] | .errorCount') -ne 0 ]]; then
|
||||||
echo "backup was not successfull"
|
echo "backup was not successful"
|
||||||
exit 1
|
exit 1
|
||||||
fi
|
fi
|
||||||
|
|
||||||
data=$( echo $resultjson | jq -r '.[0] | .id' )
|
data=$( echo $resultjson | jq -r '.[0] | .id' )
|
||||||
echo result=$data >> $GITHUB_OUTPUT
|
echo result=$data >> $GITHUB_OUTPUT
|
||||||
|
|
||||||
# list the bakcup onedrive
|
# list all onedrive backups
|
||||||
- name: Backup onedrive list test
|
- name: Backup onedrive list test
|
||||||
run: |
|
run: |
|
||||||
set -euo pipefail
|
set -euo pipefail
|
||||||
./corso backup list onedrive \
|
./corso backup list onedrive \
|
||||||
--hide-progress 2>&1 | tee $TEST_RESULT/backup_onedrive_list.txt
|
--hide-progress \
|
||||||
|
2>&1 | tee $TEST_RESULT/backup_onedrive_list.txt
|
||||||
|
|
||||||
if ! grep -q ${{ steps.onedrive-test.outputs.result }} $TEST_RESULT/backup_onedrive_list.txt
|
if ! grep -q ${{ steps.onedrive-test.outputs.result }} $TEST_RESULT/backup_onedrive_list.txt
|
||||||
then
|
then
|
||||||
echo "listing of backup was not successfull"
|
echo "listing of backup was not successful"
|
||||||
exit 1
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
# list the previous onedrive backup
|
||||||
|
- name: Backup onedrive list test
|
||||||
|
run: |
|
||||||
|
set -euo pipefail
|
||||||
|
./corso backup list onedrive \
|
||||||
|
--hide-progress \
|
||||||
|
--backup "${{ steps.onedrive-test.outputs.result }}" \
|
||||||
|
2>&1 | tee $TEST_RESULT/backup_onedrive_list_single.txt
|
||||||
|
|
||||||
|
if ! grep -q ${{ steps.onedrive-test.outputs.result }} $TEST_RESULT/backup_onedrive_list.txt
|
||||||
|
then
|
||||||
|
echo "listing of backup was not successful"
|
||||||
|
exit 1
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# test onedrive restore
|
# test onedrive restore
|
||||||
@ -222,12 +262,16 @@ jobs:
|
|||||||
id: onedrive-restore-test
|
id: onedrive-restore-test
|
||||||
run: |
|
run: |
|
||||||
set -euo pipefail
|
set -euo pipefail
|
||||||
./corso restore onedrive --backup "${{ steps.onedrive-test.outputs.result }}" --hide-progress 2>&1 | tee $TEST_RESULT/onedrive-restore-test.txt
|
./corso restore onedrive \
|
||||||
|
--hide-progress \
|
||||||
|
--backup "${{ steps.onedrive-test.outputs.result }}" \
|
||||||
|
2>&1 | tee $TEST_RESULT/onedrive-restore-test.txt
|
||||||
echo result=$(grep -i -e 'Restoring to folder ' $TEST_RESULT/onedrive-restore-test.txt | sed "s/Restoring to folder//") >> $GITHUB_OUTPUT
|
echo result=$(grep -i -e 'Restoring to folder ' $TEST_RESULT/onedrive-restore-test.txt | sed "s/Restoring to folder//") >> $GITHUB_OUTPUT
|
||||||
|
|
||||||
- name: Restoration oneDrive check
|
- name: Restoration oneDrive check
|
||||||
env:
|
env:
|
||||||
RESTORE_FOLDER: ${{ steps.onedrive-restore-test.outputs.result }}
|
SANITY_RESTORE_FOLDER: ${{ steps.onedrive-restore-test.outputs.result }}
|
||||||
|
SANITY_RESTORE_SERVICE: "onedrive"
|
||||||
run: |
|
run: |
|
||||||
set -euo pipefail
|
set -euo pipefail
|
||||||
./sanityCheck
|
./sanityCheck
|
||||||
@ -238,14 +282,16 @@ jobs:
|
|||||||
run: |
|
run: |
|
||||||
set -euo pipefail
|
set -euo pipefail
|
||||||
./corso backup create onedrive \
|
./corso backup create onedrive \
|
||||||
|
--hide-progress \
|
||||||
--user "${CORSO_M365_TEST_USER_ID}"\
|
--user "${CORSO_M365_TEST_USER_ID}"\
|
||||||
--hide-progress --json 2>&1 | tee $TEST_RESULT/backup_onedrive_incremental.txt
|
--json \
|
||||||
|
2>&1 | tee $TEST_RESULT/backup_onedrive_incremental.txt
|
||||||
|
|
||||||
resultjson=$(sed -e '1,/Completed Backups/d' $TEST_RESULT/backup_onedrive_incremental.txt )
|
resultjson=$(sed -e '1,/Completed Backups/d' $TEST_RESULT/backup_onedrive_incremental.txt )
|
||||||
|
|
||||||
if [[ $( echo $resultjson | jq -r '.[0] | .errorCount') -ne 0 ]]; then
|
if [[ $( echo $resultjson | jq -r '.[0] | .errorCount') -ne 0 ]]; then
|
||||||
echo "backup was not successfull"
|
echo "backup was not successful"
|
||||||
exit 1
|
exit 1
|
||||||
fi
|
fi
|
||||||
|
|
||||||
data=$( echo $resultjson | jq -r '.[0] | .id' )
|
data=$( echo $resultjson | jq -r '.[0] | .id' )
|
||||||
@ -256,13 +302,27 @@ jobs:
|
|||||||
id: onedrive-incremental-restore-test
|
id: onedrive-incremental-restore-test
|
||||||
run: |
|
run: |
|
||||||
set -euo pipefail
|
set -euo pipefail
|
||||||
./corso restore onedrive --backup "${{ steps.onedrive-incremental-test.outputs.result }}" --hide-progress 2>&1 | tee $TEST_RESULT/onedrive-incremental-restore-test.txt
|
./corso restore onedrive \
|
||||||
|
--hide-progress \
|
||||||
|
--backup "${{ steps.onedrive-incremental-test.outputs.result }}" \
|
||||||
|
2>&1 | tee $TEST_RESULT/onedrive-incremental-restore-test.txt
|
||||||
echo result=$(grep -i -e 'Restoring to folder ' $TEST_RESULT/onedrive-incremental-restore-test.txt | sed "s/Restoring to folder//") >> $GITHUB_OUTPUT
|
echo result=$(grep -i -e 'Restoring to folder ' $TEST_RESULT/onedrive-incremental-restore-test.txt | sed "s/Restoring to folder//") >> $GITHUB_OUTPUT
|
||||||
|
|
||||||
- name: Restoration oneDrive check
|
- name: Restoration oneDrive check
|
||||||
env:
|
env:
|
||||||
RESTORE_FOLDER: ${{ steps.onedrive-incremental-restore-test.outputs.result }}
|
SANITY_RESTORE_FOLDER: ${{ steps.onedrive-incremental-restore-test.outputs.result }}
|
||||||
|
SANITY_RESTORE_SERVICE: "onedrive"
|
||||||
run: |
|
run: |
|
||||||
set -euo pipefail
|
set -euo pipefail
|
||||||
./sanityCheck
|
./sanityCheck
|
||||||
|
|
||||||
|
# Upload the original go test output as an artifact for later review.
|
||||||
|
- name: Upload test log
|
||||||
|
if: failure()
|
||||||
|
uses: actions/upload-artifact@v3
|
||||||
|
with:
|
||||||
|
name: test-log
|
||||||
|
path: src/testlog/*
|
||||||
|
if-no-files-found: error
|
||||||
|
retention-days: 14
|
||||||
|
|
||||||
@ -2,99 +2,112 @@ package main
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
"os"
|
"os"
|
||||||
"reflect"
|
"path"
|
||||||
"strings"
|
"strings"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/alcionai/corso/src/internal/common/ptr"
|
"github.com/alcionai/clues"
|
||||||
"github.com/alcionai/corso/src/internal/connector/graph"
|
|
||||||
msgraphsdk "github.com/microsoftgraph/msgraph-sdk-go"
|
msgraphsdk "github.com/microsoftgraph/msgraph-sdk-go"
|
||||||
"github.com/microsoftgraph/msgraph-sdk-go/models"
|
"github.com/microsoftgraph/msgraph-sdk-go/models"
|
||||||
"github.com/microsoftgraph/msgraph-sdk-go/users"
|
"github.com/microsoftgraph/msgraph-sdk-go/users"
|
||||||
|
"golang.org/x/exp/maps"
|
||||||
|
"golang.org/x/exp/slices"
|
||||||
|
|
||||||
|
"github.com/alcionai/corso/src/internal/common"
|
||||||
|
"github.com/alcionai/corso/src/internal/common/ptr"
|
||||||
|
"github.com/alcionai/corso/src/internal/connector/graph"
|
||||||
|
"github.com/alcionai/corso/src/pkg/logger"
|
||||||
)
|
)
|
||||||
|
|
||||||
func main() {
|
func main() {
|
||||||
|
ctx, log := logger.Seed(context.Background(), "info", logger.GetLogFile(""))
|
||||||
|
defer func() {
|
||||||
|
_ = log.Sync() // flush all logs in the buffer
|
||||||
|
}()
|
||||||
|
|
||||||
adapter, err := graph.CreateAdapter(
|
adapter, err := graph.CreateAdapter(
|
||||||
os.Getenv("AZURE_TENANT_ID"),
|
os.Getenv("AZURE_TENANT_ID"),
|
||||||
os.Getenv("AZURE_CLIENT_ID"),
|
os.Getenv("AZURE_CLIENT_ID"),
|
||||||
os.Getenv("AZURE_CLIENT_SECRET"))
|
os.Getenv("AZURE_CLIENT_SECRET"))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fmt.Println("error while creating adapter: ", err)
|
fatal(ctx, "creating adapter", err)
|
||||||
os.Exit(1)
|
|
||||||
|
|
||||||
return
|
|
||||||
}
|
}
|
||||||
|
|
||||||
testUser := os.Getenv("CORSO_M365_TEST_USER_ID")
|
var (
|
||||||
folder := strings.TrimSpace(os.Getenv("RESTORE_FOLDER"))
|
client = msgraphsdk.NewGraphServiceClient(adapter)
|
||||||
|
testUser = os.Getenv("CORSO_M365_TEST_USER_ID")
|
||||||
|
testService = os.Getenv("SANITY_RESTORE_SERVICE")
|
||||||
|
folder = strings.TrimSpace(os.Getenv("SANITY_RESTORE_FOLDER"))
|
||||||
|
startTime, _ = mustGetTimeFromName(ctx, folder)
|
||||||
|
)
|
||||||
|
|
||||||
restoreStartTime := strings.SplitAfter(folder, "Corso_Restore_")[1]
|
ctx = clues.Add(
|
||||||
startTime, _ := time.Parse(time.RFC822, restoreStartTime)
|
ctx,
|
||||||
|
"resource_owner", testUser,
|
||||||
|
"service", testService,
|
||||||
|
"sanity_restore_folder", folder,
|
||||||
|
"start_time", startTime.Format(time.RFC3339Nano))
|
||||||
|
|
||||||
fmt.Println("Restore folder: ", folder)
|
logger.Ctx(ctx).Info("starting sanity test check")
|
||||||
|
|
||||||
client := msgraphsdk.NewGraphServiceClient(adapter)
|
switch testService {
|
||||||
|
|
||||||
switch service := os.Getenv("RESTORE_SERVICE"); service {
|
|
||||||
case "exchange":
|
case "exchange":
|
||||||
checkEmailRestoration(client, testUser, folder, startTime)
|
checkEmailRestoration(ctx, client, testUser, folder, startTime)
|
||||||
|
case "onedrive":
|
||||||
|
checkOnedriveRestoration(ctx, client, testUser, folder, startTime)
|
||||||
default:
|
default:
|
||||||
checkOnedriveRestoration(client, testUser, folder, startTime)
|
fatal(ctx, "no service specified", nil)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// checkEmailRestoration verifies that the emails count in restored folder is equivalent to
|
// checkEmailRestoration verifies that the emails count in restored folder is equivalent to
|
||||||
// emails in actual m365 account
|
// emails in actual m365 account
|
||||||
func checkEmailRestoration(
|
func checkEmailRestoration(
|
||||||
|
ctx context.Context,
|
||||||
client *msgraphsdk.GraphServiceClient,
|
client *msgraphsdk.GraphServiceClient,
|
||||||
testUser,
|
testUser, folderName string,
|
||||||
folderName string,
|
|
||||||
startTime time.Time,
|
startTime time.Time,
|
||||||
) {
|
) {
|
||||||
var (
|
var (
|
||||||
messageCount = make(map[string]int32)
|
itemCount = make(map[string]int32)
|
||||||
restoreFolder models.MailFolderable
|
restoreFolder models.MailFolderable
|
||||||
|
builder = client.UsersById(testUser).MailFolders()
|
||||||
)
|
)
|
||||||
|
|
||||||
user := client.UsersById(testUser)
|
|
||||||
builder := user.MailFolders()
|
|
||||||
|
|
||||||
for {
|
for {
|
||||||
result, err := builder.Get(context.Background(), nil)
|
result, err := builder.Get(ctx, nil)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fmt.Printf("Error getting the drive: %v\n", err)
|
fatal(ctx, "getting mail folders", err)
|
||||||
os.Exit(1)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
res := result.GetValue()
|
values := result.GetValue()
|
||||||
|
|
||||||
for _, r := range res {
|
// recursive restore folder discovery before proceeding with tests
|
||||||
name, ok := ptr.ValOK(r.GetDisplayName())
|
for _, v := range values {
|
||||||
if !ok {
|
var (
|
||||||
|
itemID = ptr.Val(v.GetId())
|
||||||
|
itemName = ptr.Val(v.GetDisplayName())
|
||||||
|
ictx = clues.Add(ctx, "item_id", itemID, "item_name", itemName)
|
||||||
|
folderTime, hasTime = mustGetTimeFromName(ctx, itemName)
|
||||||
|
)
|
||||||
|
|
||||||
|
if !isWithinTimeBound(ictx, startTime, folderTime, hasTime) {
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
var rStartTime time.Time
|
// if we found the folder to testt against, back out of this loop.
|
||||||
|
if itemName == folderName {
|
||||||
restoreStartTime := strings.SplitAfter(name, "Corso_Restore_")
|
restoreFolder = v
|
||||||
if len(restoreStartTime) > 1 {
|
|
||||||
rStartTime, _ = time.Parse(time.RFC822, restoreStartTime[1])
|
|
||||||
if startTime.Before(rStartTime) {
|
|
||||||
fmt.Printf("The restore folder %s was created after %s. Will skip check.", name, folderName)
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if name == folderName {
|
|
||||||
restoreFolder = r
|
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
getAllSubFolder(client, testUser, r, name, messageCount)
|
// otherwise, recursively aggregate all child folders.
|
||||||
|
getAllSubFolder(ctx, client, testUser, v, itemName, itemCount)
|
||||||
|
|
||||||
messageCount[name], _ = ptr.ValOK(r.GetTotalItemCount())
|
itemCount[itemName] = ptr.Val(v.GetTotalItemCount())
|
||||||
}
|
}
|
||||||
|
|
||||||
link, ok := ptr.ValOK(result.GetOdataNextLink())
|
link, ok := ptr.ValOK(result.GetOdataNextLink())
|
||||||
@ -105,86 +118,93 @@ func checkEmailRestoration(
|
|||||||
builder = users.NewItemMailFoldersRequestBuilder(link, client.GetAdapter())
|
builder = users.NewItemMailFoldersRequestBuilder(link, client.GetAdapter())
|
||||||
}
|
}
|
||||||
|
|
||||||
folderID, ok := ptr.ValOK(restoreFolder.GetId())
|
folderID := ptr.Val(restoreFolder.GetId())
|
||||||
if !ok {
|
folderName = ptr.Val(restoreFolder.GetDisplayName())
|
||||||
fmt.Printf("can't find ID of restore folder")
|
ctx = clues.Add(
|
||||||
os.Exit(1)
|
ctx,
|
||||||
}
|
"restore_folder_id", folderID,
|
||||||
|
"restore_folder_name", folderName)
|
||||||
|
|
||||||
folder := user.MailFoldersById(folderID)
|
childFolder, err := client.
|
||||||
|
UsersById(testUser).
|
||||||
childFolder, err := folder.ChildFolders().Get(context.Background(), nil)
|
MailFoldersById(folderID).
|
||||||
|
ChildFolders().
|
||||||
|
Get(ctx, nil)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fmt.Printf("Error getting the drive: %v\n", err)
|
fatal(ctx, "getting restore folder child folders", err)
|
||||||
os.Exit(1)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
for _, restore := range childFolder.GetValue() {
|
for _, fld := range childFolder.GetValue() {
|
||||||
restoreDisplayName, ok := ptr.ValOK(restore.GetDisplayName())
|
var (
|
||||||
if !ok {
|
fldID = ptr.Val(fld.GetId())
|
||||||
continue
|
fldName = ptr.Val(fld.GetDisplayName())
|
||||||
}
|
count = ptr.Val(fld.GetTotalItemCount())
|
||||||
|
ictx = clues.Add(
|
||||||
|
ctx,
|
||||||
|
"child_folder_id", fldID,
|
||||||
|
"child_folder_name", fldName,
|
||||||
|
"expected_count", itemCount[fldName],
|
||||||
|
"actual_count", count)
|
||||||
|
)
|
||||||
|
|
||||||
restoreItemCount, _ := ptr.ValOK(restore.GetTotalItemCount())
|
if itemCount[fldName] != count {
|
||||||
|
logger.Ctx(ictx).Error("test failure: Restore item counts do not match")
|
||||||
if messageCount[restoreDisplayName] != restoreItemCount {
|
fmt.Println("Restore item counts do not match:")
|
||||||
fmt.Println("Restore was not succesfull for: ", restoreDisplayName,
|
fmt.Println("* expected:", itemCount[fldName])
|
||||||
"Folder count: ", messageCount[restoreDisplayName],
|
fmt.Println("* actual:", count)
|
||||||
"Restore count: ", restoreItemCount)
|
fmt.Println("Folder:", fldName, ptr.Val(fld.GetId()))
|
||||||
os.Exit(1)
|
os.Exit(1)
|
||||||
}
|
}
|
||||||
|
|
||||||
checkAllSubFolder(client, testUser, restore, restoreDisplayName, messageCount)
|
checkAllSubFolder(ctx, client, testUser, fld, fldName, itemCount)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// getAllSubFolder will recursively check for all subfolders and get the corresponding
|
// getAllSubFolder will recursively check for all subfolders and get the corresponding
|
||||||
// email count.
|
// email count.
|
||||||
func getAllSubFolder(
|
func getAllSubFolder(
|
||||||
|
ctx context.Context,
|
||||||
client *msgraphsdk.GraphServiceClient,
|
client *msgraphsdk.GraphServiceClient,
|
||||||
testUser string,
|
testUser string,
|
||||||
r models.MailFolderable,
|
r models.MailFolderable,
|
||||||
parentFolder string,
|
parentFolder string,
|
||||||
messageCount map[string]int32,
|
messageCount map[string]int32,
|
||||||
) {
|
) {
|
||||||
folderID, ok := ptr.ValOK(r.GetId())
|
var (
|
||||||
|
folderID = ptr.Val(r.GetId())
|
||||||
if !ok {
|
count int32 = 99
|
||||||
fmt.Println("unable to get sub folder ID")
|
options = &users.ItemMailFoldersItemChildFoldersRequestBuilderGetRequestConfiguration{
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
user := client.UsersById(testUser)
|
|
||||||
folder := user.MailFoldersById(folderID)
|
|
||||||
|
|
||||||
var count int32 = 99
|
|
||||||
|
|
||||||
childFolder, err := folder.ChildFolders().Get(
|
|
||||||
context.Background(),
|
|
||||||
&users.ItemMailFoldersItemChildFoldersRequestBuilderGetRequestConfiguration{
|
|
||||||
QueryParameters: &users.ItemMailFoldersItemChildFoldersRequestBuilderGetQueryParameters{
|
QueryParameters: &users.ItemMailFoldersItemChildFoldersRequestBuilderGetQueryParameters{
|
||||||
Top: &count,
|
Top: &count,
|
||||||
},
|
},
|
||||||
})
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
ctx = clues.Add(ctx, "parent_folder_id", folderID)
|
||||||
|
|
||||||
|
childFolder, err := client.
|
||||||
|
UsersById(testUser).
|
||||||
|
MailFoldersById(folderID).
|
||||||
|
ChildFolders().
|
||||||
|
Get(ctx, options)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fmt.Printf("Error getting the drive: %v\n", err)
|
fatal(ctx, "getting mail subfolders", err)
|
||||||
os.Exit(1)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
for _, child := range childFolder.GetValue() {
|
for _, child := range childFolder.GetValue() {
|
||||||
childDisplayName, _ := ptr.ValOK(child.GetDisplayName())
|
var (
|
||||||
|
childDisplayName = ptr.Val(child.GetDisplayName())
|
||||||
fullFolderName := parentFolder + "/" + childDisplayName
|
childFolderCount = ptr.Val(child.GetChildFolderCount())
|
||||||
|
fullFolderName = parentFolder + "/" + childDisplayName
|
||||||
|
)
|
||||||
|
|
||||||
messageCount[fullFolderName], _ = ptr.ValOK(child.GetTotalItemCount())
|
messageCount[fullFolderName], _ = ptr.ValOK(child.GetTotalItemCount())
|
||||||
|
|
||||||
childFolderCount, _ := ptr.ValOK(child.GetChildFolderCount())
|
|
||||||
|
|
||||||
// recursively check for subfolders
|
// recursively check for subfolders
|
||||||
if childFolderCount > 0 {
|
if childFolderCount > 0 {
|
||||||
parentFolder := fullFolderName
|
parentFolder := fullFolderName
|
||||||
|
|
||||||
getAllSubFolder(client, testUser, child, parentFolder, messageCount)
|
getAllSubFolder(ctx, client, testUser, child, parentFolder, messageCount)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -192,173 +212,251 @@ func getAllSubFolder(
|
|||||||
// checkAllSubFolder will recursively traverse inside the restore folder and
|
// checkAllSubFolder will recursively traverse inside the restore folder and
|
||||||
// verify that data matched in all subfolders
|
// verify that data matched in all subfolders
|
||||||
func checkAllSubFolder(
|
func checkAllSubFolder(
|
||||||
|
ctx context.Context,
|
||||||
client *msgraphsdk.GraphServiceClient,
|
client *msgraphsdk.GraphServiceClient,
|
||||||
testUser string,
|
testUser string,
|
||||||
r models.MailFolderable,
|
r models.MailFolderable,
|
||||||
parentFolder string,
|
parentFolder string,
|
||||||
messageCount map[string]int32,
|
messageCount map[string]int32,
|
||||||
) {
|
) {
|
||||||
folderID, ok := ptr.ValOK(r.GetId())
|
var (
|
||||||
|
folderID = ptr.Val(r.GetId())
|
||||||
if !ok {
|
count int32 = 99
|
||||||
fmt.Println("unable to get sub folder ID")
|
options = &users.ItemMailFoldersItemChildFoldersRequestBuilderGetRequestConfiguration{
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
user := client.UsersById(testUser)
|
|
||||||
folder := user.MailFoldersById(folderID)
|
|
||||||
|
|
||||||
var count int32 = 99
|
|
||||||
|
|
||||||
childFolder, err := folder.ChildFolders().Get(
|
|
||||||
context.Background(),
|
|
||||||
&users.ItemMailFoldersItemChildFoldersRequestBuilderGetRequestConfiguration{
|
|
||||||
QueryParameters: &users.ItemMailFoldersItemChildFoldersRequestBuilderGetQueryParameters{
|
QueryParameters: &users.ItemMailFoldersItemChildFoldersRequestBuilderGetQueryParameters{
|
||||||
Top: &count,
|
Top: &count,
|
||||||
},
|
},
|
||||||
})
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
childFolder, err := client.
|
||||||
|
UsersById(testUser).
|
||||||
|
MailFoldersById(folderID).
|
||||||
|
ChildFolders().
|
||||||
|
Get(ctx, options)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fmt.Printf("Error getting the drive: %v\n", err)
|
fatal(ctx, "getting mail subfolders", err)
|
||||||
os.Exit(1)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
for _, child := range childFolder.GetValue() {
|
for _, child := range childFolder.GetValue() {
|
||||||
childDisplayName, _ := ptr.ValOK(child.GetDisplayName())
|
var (
|
||||||
|
childDisplayName = ptr.Val(child.GetDisplayName())
|
||||||
fullFolderName := parentFolder + "/" + childDisplayName
|
childTotalCount = ptr.Val(child.GetTotalItemCount())
|
||||||
|
//nolint:forbidigo
|
||||||
childTotalCount, _ := ptr.ValOK(child.GetTotalItemCount())
|
fullFolderName = path.Join(parentFolder, childDisplayName)
|
||||||
|
)
|
||||||
|
|
||||||
if messageCount[fullFolderName] != childTotalCount {
|
if messageCount[fullFolderName] != childTotalCount {
|
||||||
fmt.Println("Restore was not succesfull for: ", fullFolderName,
|
fmt.Println("Message count doesn't match:")
|
||||||
"Folder count: ", messageCount[fullFolderName],
|
fmt.Println("* expected:", messageCount[fullFolderName])
|
||||||
"Restore count: ", childTotalCount)
|
fmt.Println("* actual:", childTotalCount)
|
||||||
|
fmt.Println("Item:", fullFolderName, folderID)
|
||||||
os.Exit(1)
|
os.Exit(1)
|
||||||
}
|
}
|
||||||
|
|
||||||
childFolderCount, _ := ptr.ValOK(child.GetChildFolderCount())
|
childFolderCount := ptr.Val(child.GetChildFolderCount())
|
||||||
|
|
||||||
if childFolderCount > 0 {
|
if childFolderCount > 0 {
|
||||||
parentFolder := fullFolderName
|
checkAllSubFolder(ctx, client, testUser, child, fullFolderName, messageCount)
|
||||||
|
|
||||||
checkAllSubFolder(client, testUser, child, parentFolder, messageCount)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func checkOnedriveRestoration(client *msgraphsdk.GraphServiceClient, testUser, folderName string, startTime time.Time) {
|
func checkOnedriveRestoration(
|
||||||
file := make(map[string]int64)
|
ctx context.Context,
|
||||||
folderPermission := make(map[string][]string)
|
client *msgraphsdk.GraphServiceClient,
|
||||||
restoreFolderID := ""
|
testUser, folderName string,
|
||||||
|
startTime time.Time,
|
||||||
|
) {
|
||||||
|
var (
|
||||||
|
// map itemID -> item size
|
||||||
|
fileSizes = make(map[string]int64)
|
||||||
|
// map itemID -> permission id -> []permission roles
|
||||||
|
folderPermission = make(map[string]map[string][]string)
|
||||||
|
)
|
||||||
|
|
||||||
drive, err := client.UsersById(testUser).Drive().Get(context.Background(), nil)
|
drive, err := client.
|
||||||
|
UsersById(testUser).
|
||||||
|
Drive().
|
||||||
|
Get(ctx, nil)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fmt.Printf("Error getting the drive: %v\n", err)
|
fatal(ctx, "getting the drive:", err)
|
||||||
os.Exit(1)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
response, err := client.DrivesById(*drive.GetId()).Root().Children().Get(context.Background(), nil)
|
var (
|
||||||
|
driveID = ptr.Val(drive.GetId())
|
||||||
|
driveName = ptr.Val(drive.GetName())
|
||||||
|
restoreFolderID string
|
||||||
|
)
|
||||||
|
|
||||||
|
ctx = clues.Add(ctx, "drive_id", driveID, "drive_name", driveName)
|
||||||
|
|
||||||
|
response, err := client.
|
||||||
|
DrivesById(driveID).
|
||||||
|
Root().
|
||||||
|
Children().
|
||||||
|
Get(ctx, nil)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fmt.Printf("Error getting drive by id: %v\n", err)
|
fatal(ctx, "getting drive by id", err)
|
||||||
os.Exit(1)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
for _, driveItem := range response.GetValue() {
|
for _, driveItem := range response.GetValue() {
|
||||||
if *driveItem.GetName() == folderName {
|
var (
|
||||||
restoreFolderID = *driveItem.GetId()
|
itemID = ptr.Val(driveItem.GetId())
|
||||||
|
itemName = ptr.Val(driveItem.GetName())
|
||||||
|
ictx = clues.Add(ctx, "item_id", itemID, "item_name", itemName)
|
||||||
|
)
|
||||||
|
|
||||||
|
if itemName == folderName {
|
||||||
|
restoreFolderID = itemID
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
var rStartTime time.Time
|
folderTime, hasTime := mustGetTimeFromName(ictx, itemName)
|
||||||
|
|
||||||
restoreStartTime := strings.SplitAfter(*driveItem.GetName(), "Corso_Restore_")
|
if !isWithinTimeBound(ctx, startTime, folderTime, hasTime) {
|
||||||
if len(restoreStartTime) > 1 {
|
continue
|
||||||
rStartTime, _ = time.Parse(time.RFC822, restoreStartTime[1])
|
|
||||||
if startTime.Before(rStartTime) {
|
|
||||||
fmt.Printf("The restore folder %s was created after %s. Will skip check.", *driveItem.GetName(), folderName)
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// if it's a file check the size
|
// if it's a file check the size
|
||||||
if driveItem.GetFile() != nil {
|
if driveItem.GetFile() != nil {
|
||||||
file[*driveItem.GetName()] = *driveItem.GetSize()
|
fileSizes[itemName] = ptr.Val(driveItem.GetSize())
|
||||||
}
|
}
|
||||||
|
|
||||||
if driveItem.GetFolder() != nil {
|
folderPermission[itemID] = permissionsIn(ctx, client, driveID, itemID, folderPermission[itemID])
|
||||||
permission, err := client.
|
|
||||||
DrivesById(*drive.GetId()).
|
|
||||||
ItemsById(*driveItem.GetId()).
|
|
||||||
Permissions().
|
|
||||||
Get(context.TODO(), nil)
|
|
||||||
if err != nil {
|
|
||||||
fmt.Printf("Error getting item by id: %v\n", err)
|
|
||||||
os.Exit(1)
|
|
||||||
}
|
|
||||||
|
|
||||||
// check if permission are correct on folder
|
|
||||||
for _, permission := range permission.GetValue() {
|
|
||||||
folderPermission[*driveItem.GetName()] = permission.GetRoles()
|
|
||||||
}
|
|
||||||
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
checkFileData(client, *drive.GetId(), restoreFolderID, file, folderPermission)
|
checkFileData(ctx, client, driveID, restoreFolderID, fileSizes, folderPermission)
|
||||||
|
|
||||||
fmt.Println("Success")
|
fmt.Println("Success")
|
||||||
}
|
}
|
||||||
|
|
||||||
func checkFileData(
|
func checkFileData(
|
||||||
|
ctx context.Context,
|
||||||
client *msgraphsdk.GraphServiceClient,
|
client *msgraphsdk.GraphServiceClient,
|
||||||
driveID,
|
driveID,
|
||||||
restoreFolderID string,
|
restoreFolderID string,
|
||||||
file map[string]int64,
|
fileSizes map[string]int64,
|
||||||
folderPermission map[string][]string,
|
folderPermission map[string]map[string][]string,
|
||||||
) {
|
) {
|
||||||
itemBuilder := client.DrivesById(driveID).ItemsById(restoreFolderID)
|
restored, err := client.
|
||||||
|
DrivesById(driveID).
|
||||||
restoreResponses, err := itemBuilder.Children().Get(context.Background(), nil)
|
ItemsById(restoreFolderID).
|
||||||
|
Children().
|
||||||
|
Get(ctx, nil)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fmt.Printf("Error getting child folder: %v\n", err)
|
fatal(ctx, "getting child folder", err)
|
||||||
os.Exit(1)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
for _, restoreData := range restoreResponses.GetValue() {
|
for _, item := range restored.GetValue() {
|
||||||
restoreName := *restoreData.GetName()
|
var (
|
||||||
|
itemID = ptr.Val(item.GetId())
|
||||||
|
itemName = ptr.Val(item.GetName())
|
||||||
|
itemSize = ptr.Val(item.GetSize())
|
||||||
|
)
|
||||||
|
|
||||||
if restoreData.GetFile() != nil {
|
if item.GetFile() != nil {
|
||||||
if *restoreData.GetSize() != file[restoreName] {
|
if itemSize != fileSizes[itemName] {
|
||||||
fmt.Printf("Size of file %s is different in drive %d and restored file: %d ",
|
fmt.Println("File size does not match:")
|
||||||
restoreName,
|
fmt.Println("* expected:", fileSizes[itemName])
|
||||||
file[restoreName],
|
fmt.Println("* actual:", itemSize)
|
||||||
*restoreData.GetSize())
|
fmt.Println("Item:", itemName, itemID)
|
||||||
os.Exit(1)
|
os.Exit(1)
|
||||||
}
|
}
|
||||||
|
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
itemBuilder := client.DrivesById(driveID).ItemsById(*restoreData.GetId())
|
if item.GetFolder() == nil && item.GetPackage() == nil {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
if restoreData.GetFolder() != nil {
|
var (
|
||||||
permissionColl, err := itemBuilder.Permissions().Get(context.TODO(), nil)
|
expectItem = folderPermission[itemID]
|
||||||
if err != nil {
|
results = permissionsIn(ctx, client, driveID, itemID, nil)
|
||||||
fmt.Printf("Error getting permission: %v\n", err)
|
)
|
||||||
os.Exit(1)
|
|
||||||
}
|
|
||||||
|
|
||||||
userPermission := []string{}
|
for pid, result := range results {
|
||||||
|
expect := expectItem[pid]
|
||||||
|
|
||||||
for _, perm := range permissionColl.GetValue() {
|
if !slices.Equal(expect, result) {
|
||||||
userPermission = perm.GetRoles()
|
fmt.Println("permissions are not equal")
|
||||||
}
|
fmt.Println("* expected: ", expect)
|
||||||
|
fmt.Println("* actual: ", result)
|
||||||
if !reflect.DeepEqual(folderPermission[restoreName], userPermission) {
|
fmt.Println("Item:", itemName, itemID)
|
||||||
fmt.Printf("Permission mismatch for %s.", restoreName)
|
fmt.Println("Permission:", pid)
|
||||||
os.Exit(1)
|
os.Exit(1)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// ---------------------------------------------------------------------------
|
||||||
|
// Helpers
|
||||||
|
// ---------------------------------------------------------------------------
|
||||||
|
|
||||||
|
func fatal(ctx context.Context, msg string, err error) {
|
||||||
|
logger.CtxErr(ctx, err).Error("test failure: " + msg)
|
||||||
|
fmt.Println(msg+": ", err)
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
|
||||||
|
func permissionsIn(
|
||||||
|
ctx context.Context,
|
||||||
|
client *msgraphsdk.GraphServiceClient,
|
||||||
|
driveID, itemID string,
|
||||||
|
init map[string][]string,
|
||||||
|
) map[string][]string {
|
||||||
|
result := map[string][]string{}
|
||||||
|
|
||||||
|
pcr, err := client.
|
||||||
|
DrivesById(driveID).
|
||||||
|
ItemsById(itemID).
|
||||||
|
Permissions().
|
||||||
|
Get(ctx, nil)
|
||||||
|
if err != nil {
|
||||||
|
fatal(ctx, "getting permission", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(init) > 0 {
|
||||||
|
maps.Copy(result, init)
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, p := range pcr.GetValue() {
|
||||||
|
var (
|
||||||
|
pid = ptr.Val(p.GetId())
|
||||||
|
roles = p.GetRoles()
|
||||||
|
)
|
||||||
|
|
||||||
|
slices.Sort(roles)
|
||||||
|
|
||||||
|
result[pid] = roles
|
||||||
|
}
|
||||||
|
|
||||||
|
return result
|
||||||
|
}
|
||||||
|
|
||||||
|
func mustGetTimeFromName(ctx context.Context, name string) (time.Time, bool) {
|
||||||
|
t, err := common.ExtractTime(name)
|
||||||
|
if err != nil && !errors.Is(err, common.ErrNoTimeString) {
|
||||||
|
fatal(ctx, "extracting time from name: "+name, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return t, !errors.Is(err, common.ErrNoTimeString)
|
||||||
|
}
|
||||||
|
|
||||||
|
func isWithinTimeBound(ctx context.Context, bound, check time.Time, skip bool) bool {
|
||||||
|
if skip {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
if bound.Before(check) {
|
||||||
|
logger.Ctx(ctx).
|
||||||
|
With("boundary_time", bound, "check_time", check).
|
||||||
|
Info("skipping restore folder: not older than time bound")
|
||||||
|
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|||||||
@ -14,7 +14,7 @@ import (
|
|||||||
|
|
||||||
const (
|
const (
|
||||||
loglevel = "info"
|
loglevel = "info"
|
||||||
logfile = "stderr"
|
logfile = logger.Stderr
|
||||||
itemID = "item_id"
|
itemID = "item_id"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|||||||
@ -49,6 +49,11 @@ const (
|
|||||||
readableLogsFN = "readable-logs"
|
readableLogsFN = "readable-logs"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
Stderr = "stderr"
|
||||||
|
Stdout = "stdout"
|
||||||
|
)
|
||||||
|
|
||||||
// Returns the default location for writing logs
|
// Returns the default location for writing logs
|
||||||
func defaultLogLocation() string {
|
func defaultLogLocation() string {
|
||||||
return filepath.Join(userLogsDir, "corso", "logs", time.Now().UTC().Format("2006-01-02T15-04-05Z")+".log")
|
return filepath.Join(userLogsDir, "corso", "logs", time.Now().UTC().Format("2006-01-02T15-04-05Z")+".log")
|
||||||
@ -104,31 +109,40 @@ func PreloadLoggingFlags() (string, string) {
|
|||||||
|
|
||||||
// retrieve the user's preferred log file location
|
// retrieve the user's preferred log file location
|
||||||
// automatically defaults to default log location
|
// automatically defaults to default log location
|
||||||
logfile, err := fs.GetString(logFileFN)
|
lffv, err := fs.GetString(logFileFN)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return "info", dlf
|
return "info", dlf
|
||||||
}
|
}
|
||||||
|
|
||||||
|
logfile := GetLogFile(lffv)
|
||||||
|
|
||||||
|
return levelString, logfile
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetLogFile parses the log file. Uses the provided value, if populated,
|
||||||
|
// then falls back to the env var, and then defaults to stderr.
|
||||||
|
func GetLogFile(logFileFlagVal string) string {
|
||||||
|
r := logFileFlagVal
|
||||||
|
|
||||||
// if not specified, attempt to fall back to env declaration.
|
// if not specified, attempt to fall back to env declaration.
|
||||||
if len(logfile) == 0 {
|
if len(r) == 0 {
|
||||||
logfile = os.Getenv("CORSO_LOG_FILE")
|
r = os.Getenv("CORSO_LOG_FILE")
|
||||||
}
|
}
|
||||||
|
|
||||||
if logfile == "-" {
|
if r == "-" {
|
||||||
logfile = "stdout"
|
r = Stdout
|
||||||
}
|
}
|
||||||
|
|
||||||
if logfile != "stdout" && logfile != "stderr" {
|
if r != Stdout && r != Stderr {
|
||||||
LogFile = logfile
|
logdir := filepath.Dir(r)
|
||||||
logdir := filepath.Dir(logfile)
|
|
||||||
|
|
||||||
err := os.MkdirAll(logdir, 0o755)
|
err := os.MkdirAll(logdir, 0o755)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return "info", "stderr"
|
return Stderr
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return levelString, logfile
|
return r
|
||||||
}
|
}
|
||||||
|
|
||||||
func genLogger(level logLevel, logfile string) (*zapcore.Core, *zap.SugaredLogger) {
|
func genLogger(level logLevel, logfile string) (*zapcore.Core, *zap.SugaredLogger) {
|
||||||
@ -183,7 +197,7 @@ func genLogger(level logLevel, logfile string) (*zapcore.Core, *zap.SugaredLogge
|
|||||||
opts = append(opts, zap.WithCaller(false))
|
opts = append(opts, zap.WithCaller(false))
|
||||||
cfg.EncoderConfig.EncodeTime = zapcore.TimeEncoderOfLayout("15:04:05.00")
|
cfg.EncoderConfig.EncodeTime = zapcore.TimeEncoderOfLayout("15:04:05.00")
|
||||||
|
|
||||||
if logfile == "stderr" || logfile == "stdout" {
|
if logfile == Stderr || logfile == Stdout {
|
||||||
cfg.EncoderConfig.EncodeLevel = zapcore.CapitalColorLevelEncoder
|
cfg.EncoderConfig.EncodeLevel = zapcore.CapitalColorLevelEncoder
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
Loading…
x
Reference in New Issue
Block a user