Troubleshooting
Overview #
Here we will cover some common issues seen during a migration — how to avoid them, how to detect them, and how to address them.
101_initial_cluster.sh in the local example. The commands in this guide also assume
you have setup the shell aliases from the example, e.g. env.sh in the local example.General and Precautionary Info #
Execute a Dry Run #
The SwitchTraffic/ReverseTraffic and Complete actions support a dry run using the --dry-run flag where no
actual steps are taken but instead the command logs all the steps that would be taken. This command will also
verify that the cluster is generally in a state where it can perform the action successfully without potentially
timing out along the way. Given that traffic cutovers can potentially cause read/write pauses or outages this can
be particularly helpful during the final cutover stage.
DDL Handling #
If you expect DDL to be executed on the source table(s) while the workflow runs and you want those DDL statements
to be replicated to the target keyspace then you will need to use one of the EXEC* options for the workflow's
on-ddl flag. Please see the
on-ddl flag documentation for additional details and
related considerations.
Running a Diff #
In most cases you should run a VDiff before switching traffic to ensure
that nothing unexpected happened which caused the data to diverge during the migration.
Performance Notes #
- VReplication workflows (including VDiff) can have a major impact on the tablet so it's recommended to use non-PRIMARY tablets whenever possible to limit any impact on production traffic- You can see the key related tablet flags and when/why you may want to set them in the VReplication tablet flag docs
- You can further control any impact on the source and target tablets using the tablet throttler
 
- VReplication workflows can generate a lot of network traffic- You should strive to keep the source and target tablets in the same cell whenever possible to limit performance and cost impacts
 
Monitoring #
It's important to properly monitor your VReplication workflows in order to detect any issues. Your primary tools for this are:
- The showsub-command (e.g.MoveTables show)
- The statussub-command (e.g.MoveTables status)
- The VReplication related metrics- Note that in most production systems the tablet endpoints would be scraped and stored in something like Prometheus where you can build dashboards and alerting on the data
 
Save Routing Rules #
The Create, SwitchTraffic/ReverseTraffic, and Cancel/Complete actions modify the
routing rules. You may want to save the routing rules before
taking an action just in case you want to restore them for any reason (note that e.g. the ReverseTraffic action
will automatically revert the routing rules):
$ vtctldclient GetRoutingRules > /tmp/routingrules.backup.json
Those can later be applied this way:
$ vtctldclient ApplyRoutingRules --rules-file=/tmp/routingrules.backup.json
Specific Errors and Issues #
Stream Never Starts #
This can be exhibited in one of two ways:
- This error is shown in the status/showaction output or theWorkflow showoutput:Error picking tablet: context has expired
- The stream never starts, which can be seen in the following ways:- The showoutput is showing an empty value in thePosfield for the stream
- The status/showaction output is showingVStream has not startedfor the stream
 
- The 
When a VReplication workflow starts or restarts the tablet selection process
runs to find a viable source tablet for the stream. The --cells and --tablet-types play a key role in this process and
if we cannot ever find a viable source tablet for the stream then you may want to expand the cells and/or tablet types
made available for the selection process.
Corrective Action #
If the workflow was only created and has not yet made any progress then you should cancel the workflow and create a new
one using different values for the --cells and --tablet-types flags. If, however, this workflow has made significant
progress that you do not wish you lose, you can update the underlying workflow record directly to modify either of those
values. For example:
$ vtctldclient MoveTables --target-keyspace customer --workflow commerce2customer status --format=json
{
  "table_copy_state": {},
  "shard_streams": {
    "customer/0": {
      "streams": [
        {
          "id": 1,
          "tablet": {
            "cell": "zone1",
            "uid": 200
          },
          "source_shard": "commerce/0",
          "position": "5d8e0b24-6873-11ee-9359-49d03ab2cdee:1-49",
          "status": "Running",
          "info": "VStream Lag: 0s; ; Tx time: Wed Oct 11 16:49:02 2023."
        }
      ]
    }
  },
  "traffic_state": "Reads Not Switched. Writes Not Switched"
}
Workflow Has SQL Errors #
We can encounter persistent SQL errors when applying replicated events on the target for a variety of reasons, but
the most common cause is incompatible DDL having been executed against the source table while the workflow is running.
You would see this error in the status/show or Workflow show output. For example:
$ vtctldclient MoveTables --target-keyspace customer --workflow commerce2customer status --format=json
{
  "table_copy_state": {},
  "shard_streams": {
    "customer/0": {
      "streams": [
        {
          "id": 1,
          "tablet": {
            "cell": "zone1",
            "uid": 200
          },
          "source_shard": "commerce/0",
          "position": "5d8e0b24-6873-11ee-9359-49d03ab2cdee:1-49",
          "status": "Error: Unknown column 'notes' in 'field list' (errno 1054) (sqlstate 42S22) during query: insert into customer(customer_id,email,notes) values (100,'test@tester.com','Lots of notes')",
          "info": "VStream Lag: 0s; ; Tx time: Wed Oct 11 16:49:02 2023."
        }
      ]
    }
  },
  "traffic_state": "Reads Not Switched. Writes Not Switched"
}
This can be caused by a DDL executed on the source table as by default — controlled by the
on-ddl flag value — DDL is ignored in the stream.
Corrective Action #
If you want the same or similar DDL to be applied on the target then you can apply that DDL on the target keyspace and then restart the workflow. For example, using the example above:
vtctldclient ApplySchema --ddl-strategy=direct --sql="alter table customer add notes varchar(100) not null" customer
vtctldclient Workflow customer.commerce2customer start
If the tables are not very large or the workflow has not made much progress, you can alternatively Cancel the current
worfklow and Create another. For example:
$ vtctldclient MoveTables --target-keyspace customer --workflow commerce2customer cancel
Successfully cancelled the commerce2customer workflow in the customer keyspace
$ vtctldclient MoveTables --target-keyspace customer --workflow commerce2customer create --source-keyspace commerce --tables 'customer,corder'
The following vreplication streams exist for workflow customer.commerce2customer:
id=2 on customer/zone1-200: Status: Running. VStream has not started.
$ vtctldclient MoveTables --target-keyspace customer --workflow commerce2customer status --format=json
{
  "table_copy_state": {},
  "shard_streams": {
    "customer/0": {
      "streams": [
        {
          "id": 2,
          "tablet": {
            "cell": "zone1",
            "uid": 200
          },
          "source_shard": "commerce/0",
          "position": "5d8e0b24-6873-11ee-9359-49d03ab2cdee:1-51",
          "status": "Running",
          "info": "VStream Lag: 0s"
        }
      ]
    }
  },
  "traffic_state": "Reads Not Switched. Writes Not Switched"
} 
Switching Traffic Fails #
You can encounter a variety of failures during the SwitchTraffic/ReverseTraffic step as a number of operations are performed. To
demonstrate that we can look at an example dry run output:
$ vtctldclient MoveTables --target-keyspace customer --workflow commerce2customer SwitchTraffic --dry-run 
SwitchTraffic dry run results for workflow customer.commerce2customer at 11 Oct 23 21:23 UTC
Lock keyspace commerce
Switch reads for tables [corder,customer] to keyspace customer for tablet types [PRIMARY,REPLICA,RDONLY]
Routing rules for tables [corder,customer] will be updated
Unlock keyspace commerce
Lock keyspace commerce
Lock keyspace customer
Stop writes on keyspace commerce for tables [corder,customer]: [keyspace:commerce;shard:0;position:MySQL56/5d8e0b24-6873-11ee-9359-49d03ab2cdee:1-51]
Wait for vreplication on stopped streams to catchup for up to 30s
Create reverse vreplication workflow commerce2customer_reverse
Create journal entries on source databases
Enable writes on keyspace customer for tables [corder,customer]
Switch routing from keyspace commerce to keyspace customer
Routing rules for tables [corder,customer] will be updated
Switch writes completed, freeze and delete vreplication streams on: [tablet:200]
Start reverse vreplication streams on: [tablet:101]
Mark vreplication streams frozen on: [keyspace:customer;shard:0;tablet:200;workflow:commerce2customer;dbname:vt_customer]
Unlock keyspace customer
Unlock keyspace commerce
disallowed due to rule: enforce denied tables #
If your queries start failing with this error then you most likely had some leftover artifacts from a previous MoveTables operation
that were not properly cleaned up by running MoveTables cancel. For
MoveTables operations, shard query serving control records (denied tables lists) are used in addition to
routing rules to ensure that all query traffic is managed by the correct keyspace
as you are often only moving some tables from one keyspace to another. If those control records are not properly cleaned up then
queries may be incorrectly denied when traffic is switched. If you e.g. were to see the following error for queries after switching
traffic for the customer table from the commerce keyspace to the customer keyspace:
code = FailedPrecondition desc = disallowed due to rule: enforce denied tables (CallerID: matt) for query SELECT * FROM customer WHERE customer_id = 1
Then you can remove those unwanted/errant denied table rules from the customer keyspace this way:
$ for type in primary replica rdonly; do
    vtctldclient SetShardTabletControl --remove customer/0 ${type}
  done
# Ensure that these changes are in place everywhere
$ vtctldclient RefreshStateByShard customer/0
Completion and Cleanup Failures #
The completion action performs a number of steps that could potentially fail. We can again use the dry run output to demonstrate the various actions that are taken:
$ vtctldclient MoveTables --target-keyspace customer --workflow commerce2customer SwitchTraffic --dry-run
SwitchTraffic dry run results for workflow customer.commerce2customer at 11 Oct 23 21:23 UTC
Lock keyspace commerce
Switch reads for tables [corder,customer] to keyspace customer for tablet types [PRIMARY,REPLICA,RDONLY]
Routing rules for tables [corder,customer] will be updated
Unlock keyspace commerce
Lock keyspace commerce
Lock keyspace customer
Stop writes on keyspace commerce for tables [corder,customer]: [keyspace:commerce;shard:0;position:MySQL56/5d8e0b24-6873-11ee-9359-49d03ab2cdee:1-51]
Wait for vreplication on stopped streams to catchup for up to 30s
Create reverse vreplication workflow commerce2customer_reverse
Create journal entries on source databases
Enable writes on keyspace customer for tables [corder,customer]
Switch routing from keyspace commerce to keyspace customer
Routing rules for tables [corder,customer] will be updated
Switch writes completed, freeze and delete vreplication streams on: [tablet:200]
Start reverse vreplication streams on: [tablet:101]
Mark vreplication streams frozen on: [keyspace:customer;shard:0;tablet:200;workflow:commerce2customer;dbname:vt_customer]
Unlock keyspace customer
Unlock keyspace commerce