Skip to content

Commit

Permalink
backupccl: split SHOW BACKUP and its tests into their own files
Browse files Browse the repository at this point in the history
  • Loading branch information
benesch committed Jun 6, 2018
1 parent 340d16f commit f85a3ce
Show file tree
Hide file tree
Showing 4 changed files with 198 additions and 155 deletions.
91 changes: 0 additions & 91 deletions pkg/ccl/backupccl/backup.go
Original file line number Diff line number Diff line change
Expand Up @@ -35,7 +35,6 @@ import (
"github.com/cockroachdb/cockroach/pkg/storage"
"github.com/cockroachdb/cockroach/pkg/storage/engine"
"github.com/cockroachdb/cockroach/pkg/util/ctxgroup"
"github.com/cockroachdb/cockroach/pkg/util/encoding"
"github.com/cockroachdb/cockroach/pkg/util/hlc"
"github.com/cockroachdb/cockroach/pkg/util/interval"
"github.com/cockroachdb/cockroach/pkg/util/log"
Expand Down Expand Up @@ -1180,95 +1179,6 @@ func backupResumeHook(typ jobs.Type, settings *cluster.Settings) jobs.Resumer {
}
}

// showBackupPlanHook implements PlanHookFn.
func showBackupPlanHook(
ctx context.Context, stmt tree.Statement, p sql.PlanHookState,
) (sql.PlanHookRowFn, sqlbase.ResultColumns, []sql.PlanNode, error) {
backup, ok := stmt.(*tree.ShowBackup)
if !ok {
return nil, nil, nil, nil
}

if err := utilccl.CheckEnterpriseEnabled(
p.ExecCfg().Settings, p.ExecCfg().ClusterID(), p.ExecCfg().Organization(), "SHOW BACKUP",
); err != nil {
return nil, nil, nil, err
}

if err := p.RequireSuperUser(ctx, "SHOW BACKUP"); err != nil {
return nil, nil, nil, err
}

toFn, err := p.TypeAsString(backup.Path, "SHOW BACKUP")
if err != nil {
return nil, nil, nil, err
}
header := sqlbase.ResultColumns{
{Name: "database", Typ: types.String},
{Name: "table", Typ: types.String},
{Name: "start_time", Typ: types.Timestamp},
{Name: "end_time", Typ: types.Timestamp},
{Name: "size_bytes", Typ: types.Int},
{Name: "rows", Typ: types.Int},
}
fn := func(ctx context.Context, _ []sql.PlanNode, resultsCh chan<- tree.Datums) error {
// TODO(dan): Move this span into sql.
ctx, span := tracing.ChildSpan(ctx, stmt.StatementTag())
defer tracing.FinishSpan(span)

str, err := toFn()
if err != nil {
return err
}
desc, err := ReadBackupDescriptorFromURI(ctx, str, p.ExecCfg().Settings)
if err != nil {
return err
}
descs := make(map[sqlbase.ID]string)
for _, descriptor := range desc.Descriptors {
if database := descriptor.GetDatabase(); database != nil {
if _, ok := descs[database.ID]; !ok {
descs[database.ID] = database.Name
}
}
}
descSizes := make(map[sqlbase.ID]roachpb.BulkOpSummary)
for _, file := range desc.Files {
// TODO(dan): This assumes each file in the backup only contains
// data from a single table, which is usually but not always
// correct. It does not account for interleaved tables or if a
// BACKUP happened to catch a newly created table that hadn't yet
// been split into its own range.
_, tableID, err := encoding.DecodeUvarintAscending(file.Span.Key)
if err != nil {
continue
}
s := descSizes[sqlbase.ID(tableID)]
s.Add(file.EntryCounts)
descSizes[sqlbase.ID(tableID)] = s
}
start := tree.DNull
if desc.StartTime.WallTime != 0 {
start = tree.MakeDTimestamp(timeutil.Unix(0, desc.StartTime.WallTime), time.Nanosecond)
}
for _, descriptor := range desc.Descriptors {
if table := descriptor.GetTable(); table != nil {
dbName := descs[table.ParentID]
resultsCh <- tree.Datums{
tree.NewDString(dbName),
tree.NewDString(table.Name),
start,
tree.MakeDTimestamp(timeutil.Unix(0, desc.EndTime.WallTime), time.Nanosecond),
tree.NewDInt(tree.DInt(descSizes[table.ID].DataSize)),
tree.NewDInt(tree.DInt(descSizes[table.ID].Rows)),
}
}
}
return nil
}
return fn, header, nil, nil
}

type versionedValues struct {
Key roachpb.Key
Values []roachpb.Value
Expand Down Expand Up @@ -1320,6 +1230,5 @@ func getAllRevisions(

func init() {
sql.AddPlanHook(backupPlanHook)
sql.AddPlanHook(showBackupPlanHook)
jobs.AddResumeHook(backupResumeHook)
}
64 changes: 0 additions & 64 deletions pkg/ccl/backupccl/backup_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -12,7 +12,6 @@ import (
"bytes"
"context"
gosql "database/sql"
"database/sql/driver"
"fmt"
"hash/crc32"
"io"
Expand Down Expand Up @@ -2448,69 +2447,6 @@ func TestRestoreDatabaseVersusTable(t *testing.T) {
})
}

func TestShowBackup(t *testing.T) {
defer leaktest.AfterTest(t)()

const numAccounts = 11
_, _, sqlDB, _, cleanupFn := backupRestoreTestSetup(t, singleNode, numAccounts, initNone)
defer cleanupFn()

full, inc := localFoo+"/full", localFoo+"/inc"

beforeFull := timeutil.Now()
sqlDB.Exec(t, `BACKUP data.bank TO $1`, full)

var unused driver.Value
var start, end *time.Time
var dataSize, rows uint64
sqlDB.QueryRow(t, `SELECT * FROM [SHOW BACKUP $1] WHERE "table" = 'bank'`, full).Scan(
&unused, &unused, &start, &end, &dataSize, &rows,
)
if start != nil {
t.Errorf("expected null start time on full backup, got %v", *start)
}
if !(*end).After(beforeFull) {
t.Errorf("expected now (%s) to be in (%s, %s)", beforeFull, start, end)
}
if dataSize <= 0 {
t.Errorf("expected dataSize to be >0 got : %d", dataSize)
}
if rows != numAccounts {
t.Errorf("expected %d got: %d", numAccounts, rows)
}

// Mess with half the rows.
affectedRows, err := sqlDB.Exec(t,
`UPDATE data.bank SET id = -1 * id WHERE id > $1`, numAccounts/2,
).RowsAffected()
if err != nil {
t.Fatal(err)
} else if affectedRows != numAccounts/2 {
t.Fatalf("expected to update %d rows, got %d", numAccounts/2, affectedRows)
}

beforeInc := timeutil.Now()
sqlDB.Exec(t, `BACKUP data.bank TO $1 INCREMENTAL FROM $2`, inc, full)

sqlDB.QueryRow(t, `SELECT * FROM [SHOW BACKUP $1] WHERE "table" = 'bank'`, inc).Scan(
&unused, &unused, &start, &end, &dataSize, &rows,
)
if start == nil {
t.Errorf("expected start time on inc backup, got %v", *start)
}
if !(*end).After(beforeInc) {
t.Errorf("expected now (%s) to be in (%s, %s)", beforeInc, start, end)
}
if dataSize <= 0 {
t.Errorf("expected dataSize to be >0 got : %d", dataSize)
}
// We added affectedRows and removed affectedRows, so there should be 2*
// affectedRows in the backup.
if expected := affectedRows * 2; rows != uint64(expected) {
t.Errorf("expected %d got: %d", expected, rows)
}
}

func TestBackupAzureAccountName(t *testing.T) {
defer leaktest.AfterTest(t)()

Expand Down
117 changes: 117 additions & 0 deletions pkg/ccl/backupccl/show.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,117 @@
// Copyright 2016 The Cockroach Authors.
//
// Licensed as a CockroachDB Enterprise file under the Cockroach Community
// License (the "License"); you may not use this file except in compliance with
// the License. You may obtain a copy of the License at
//
// https://github.com/cockroachdb/cockroach/blob/master/licenses/CCL.txt

package backupccl

import (
"context"
"time"

"github.com/cockroachdb/cockroach/pkg/ccl/utilccl"
"github.com/cockroachdb/cockroach/pkg/roachpb"
"github.com/cockroachdb/cockroach/pkg/sql"
"github.com/cockroachdb/cockroach/pkg/sql/sem/tree"
"github.com/cockroachdb/cockroach/pkg/sql/sem/types"
"github.com/cockroachdb/cockroach/pkg/sql/sqlbase"
"github.com/cockroachdb/cockroach/pkg/util/encoding"
"github.com/cockroachdb/cockroach/pkg/util/timeutil"
"github.com/cockroachdb/cockroach/pkg/util/tracing"
)

// showBackupPlanHook implements PlanHookFn.
func showBackupPlanHook(
ctx context.Context, stmt tree.Statement, p sql.PlanHookState,
) (sql.PlanHookRowFn, sqlbase.ResultColumns, []sql.PlanNode, error) {
backup, ok := stmt.(*tree.ShowBackup)
if !ok {
return nil, nil, nil, nil
}

if err := utilccl.CheckEnterpriseEnabled(
p.ExecCfg().Settings, p.ExecCfg().ClusterID(), p.ExecCfg().Organization(), "SHOW BACKUP",
); err != nil {
return nil, nil, nil, err
}

if err := p.RequireSuperUser(ctx, "SHOW BACKUP"); err != nil {
return nil, nil, nil, err
}

toFn, err := p.TypeAsString(backup.Path, "SHOW BACKUP")
if err != nil {
return nil, nil, nil, err
}
header := sqlbase.ResultColumns{
{Name: "database", Typ: types.String},
{Name: "table", Typ: types.String},
{Name: "start_time", Typ: types.Timestamp},
{Name: "end_time", Typ: types.Timestamp},
{Name: "size_bytes", Typ: types.Int},
{Name: "rows", Typ: types.Int},
}
fn := func(ctx context.Context, _ []sql.PlanNode, resultsCh chan<- tree.Datums) error {
// TODO(dan): Move this span into sql.
ctx, span := tracing.ChildSpan(ctx, stmt.StatementTag())
defer tracing.FinishSpan(span)

str, err := toFn()
if err != nil {
return err
}
desc, err := ReadBackupDescriptorFromURI(ctx, str, p.ExecCfg().Settings)
if err != nil {
return err
}
descs := make(map[sqlbase.ID]string)
for _, descriptor := range desc.Descriptors {
if database := descriptor.GetDatabase(); database != nil {
if _, ok := descs[database.ID]; !ok {
descs[database.ID] = database.Name
}
}
}
descSizes := make(map[sqlbase.ID]roachpb.BulkOpSummary)
for _, file := range desc.Files {
// TODO(dan): This assumes each file in the backup only contains
// data from a single table, which is usually but not always
// correct. It does not account for interleaved tables or if a
// BACKUP happened to catch a newly created table that hadn't yet
// been split into its own range.
_, tableID, err := encoding.DecodeUvarintAscending(file.Span.Key)
if err != nil {
continue
}
s := descSizes[sqlbase.ID(tableID)]
s.Add(file.EntryCounts)
descSizes[sqlbase.ID(tableID)] = s
}
start := tree.DNull
if desc.StartTime.WallTime != 0 {
start = tree.MakeDTimestamp(timeutil.Unix(0, desc.StartTime.WallTime), time.Nanosecond)
}
for _, descriptor := range desc.Descriptors {
if table := descriptor.GetTable(); table != nil {
dbName := descs[table.ParentID]
resultsCh <- tree.Datums{
tree.NewDString(dbName),
tree.NewDString(table.Name),
start,
tree.MakeDTimestamp(timeutil.Unix(0, desc.EndTime.WallTime), time.Nanosecond),
tree.NewDInt(tree.DInt(descSizes[table.ID].DataSize)),
tree.NewDInt(tree.DInt(descSizes[table.ID].Rows)),
}
}
}
return nil
}
return fn, header, nil, nil
}

func init() {
sql.AddPlanHook(showBackupPlanHook)
}
81 changes: 81 additions & 0 deletions pkg/ccl/backupccl/show_test.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,81 @@
// Copyright 2018 The Cockroach Authors.
//
// Licensed as a CockroachDB Enterprise file under the Cockroach Community
// License (the "License"); you may not use this file except in compliance with
// the License. You may obtain a copy of the License at
//
// https://github.com/cockroachdb/cockroach/blob/master/licenses/CCL.txt

package backupccl_test

import (
"database/sql/driver"
"testing"
"time"

"github.com/cockroachdb/cockroach/pkg/util/leaktest"
"github.com/cockroachdb/cockroach/pkg/util/timeutil"
)

func TestShowBackup(t *testing.T) {
defer leaktest.AfterTest(t)()

const numAccounts = 11
_, _, sqlDB, _, cleanupFn := backupRestoreTestSetup(t, singleNode, numAccounts, initNone)
defer cleanupFn()

full, inc := localFoo+"/full", localFoo+"/inc"

beforeFull := timeutil.Now()
sqlDB.Exec(t, `BACKUP data.bank TO $1`, full)

var unused driver.Value
var start, end *time.Time
var dataSize, rows uint64
sqlDB.QueryRow(t, `SELECT * FROM [SHOW BACKUP $1] WHERE "table" = 'bank'`, full).Scan(
&unused, &unused, &start, &end, &dataSize, &rows,
)
if start != nil {
t.Errorf("expected null start time on full backup, got %v", *start)
}
if !(*end).After(beforeFull) {
t.Errorf("expected now (%s) to be in (%s, %s)", beforeFull, start, end)
}
if dataSize <= 0 {
t.Errorf("expected dataSize to be >0 got : %d", dataSize)
}
if rows != numAccounts {
t.Errorf("expected %d got: %d", numAccounts, rows)
}

// Mess with half the rows.
affectedRows, err := sqlDB.Exec(t,
`UPDATE data.bank SET id = -1 * id WHERE id > $1`, numAccounts/2,
).RowsAffected()
if err != nil {
t.Fatal(err)
} else if affectedRows != numAccounts/2 {
t.Fatalf("expected to update %d rows, got %d", numAccounts/2, affectedRows)
}

beforeInc := timeutil.Now()
sqlDB.Exec(t, `BACKUP data.bank TO $1 INCREMENTAL FROM $2`, inc, full)

sqlDB.QueryRow(t, `SELECT * FROM [SHOW BACKUP $1] WHERE "table" = 'bank'`, inc).Scan(
&unused, &unused, &start, &end, &dataSize, &rows,
)
if start == nil {
t.Errorf("expected start time on inc backup, got %v", *start)
}
if !(*end).After(beforeInc) {
t.Errorf("expected now (%s) to be in (%s, %s)", beforeInc, start, end)
}
if dataSize <= 0 {
t.Errorf("expected dataSize to be >0 got : %d", dataSize)
}
// We added affectedRows and removed affectedRows, so there should be 2*
// affectedRows in the backup.
if expected := affectedRows * 2; rows != uint64(expected) {
t.Errorf("expected %d got: %d", expected, rows)
}
}

0 comments on commit f85a3ce

Please sign in to comment.