'Alter table modify enum in Knex js for Postgresql gives error
I am using knex js and postgresql database. I have used a migration file to create a table knex migrate:make create_car_table. In this I have added a column fuel_type. table.enu('fuel_type', ['PETROL', 'DIESEL', 'CNG']).
Now I need to alter the table and I need these enum values ['HYBRID', 'ELECTRIC', 'PETROL', 'DIESEL'].
I have created another migration file using knex migrate:make alter_car_table and added the below code
exports.up = function(knex, Promise) {
return knex.schema.alterTable('car', function (table) {
table.enu('fuel_type', ['HYBRID', 'ELECTRIC', 'PETROL', 'DIESEL']).alter();
});
};
exports.down = function(knex, Promise) {
return knex.schema.alterTable('car', function (table) {
table.enu('fuel_type', ['PETROL', 'DIESEL', 'CNG']).alter();
});
};
when I run knex migrate:latest I get the below error.
Knex:warning - migrations failed with error: alter table "car" alter column "fuel_type" type text check ("fuel_type" in ('HYBRID', 'ELECTRIC', 'PETROL', 'DIESEL')) using ("fuel_type"::text check ("fuel_type" in ('HYBRID', 'ELECTRIC', 'PETROL', 'DIESEL'))) - syntax error at or near "check"
I have refered Knex Js for this.
Solution 1:[1]
Alter column does not work for enum types in knex 0.13.0.
Also enums are implemented as check constraints, so to change it you need to recreate the.
Something like this:
exports.up = function(knex, Promise) {
return knex.schema.raw(`
ALTER TABLE "car"
DROP CONSTRAINT "car_fuel_type_check",
ADD CONSTRAINT "car_fuel_type_check"
CHECK (fuel_type IN ('HYBRID', 'ELECTRIC', 'PETROL', 'DIESEL'))
`);
};
exports.down = function(knex, Promise) { ... };
You might need to check your constraint name that was originally generated by knex from the DB.
Currently knex.schema.raw is the only way to modify enums.
Solution 2:[2]
You first need to drop the existing constraint, and create a new one with the new values. The code sample below should help.
exports.up = function(knex, Promise) {
return knex.schema.raw(`
ALTER TABLE "car" DROP CONSTRAINT "car_fuel_type_check";
ALTER TABLE "car" ADD CONSTRAINT "car_fuel_type_check" CHECK (fuel_type IN ('HYBRID'::text, 'ELECTRIC'::text, 'PETROL'::text, 'DIESEL'::text))
`);
};
// The reverse migration is similar
exports.down = function(knex, Promise) {
return knex.schema.raw(`
ALTER TABLE "car" DROP CONSTRAINT "car_fuel_type_check";
ALTER TABLE "car" ADD CONSTRAINT "car_fuel_type_check" CHECK (fuel_type IN ('PETROL'::text, 'DIESEL'::text, 'CNG'::text));
`);
};
I'm assuming your constraint name is car_fuel_type_check. If not, you should replace car_fuel_type_check with your constraint name.
Solution 3:[3]
We can update enum's value by using knex migrate.
exports.up = async function(knex) {
return knex.raw(`
ALTER TABLE organizations
CHANGE subscriptionStatus subscriptionStatus enum('Past Due','Paid','Free Trial','Free Trial Expired','Pre Trial','ddd') DEFAULT 'Pre Trial';
`);
};
exports.down = async function(knex) {
return knex.raw(`
ALTER TABLE organizations
CHANGE subscriptionStatus subscriptionStatus enum('Past Due','Paid','Free Trial','Free Trial Expired','Pre Trial') DEFAULT 'Pre Trial';
`);
};
Solution 4:[4]
As alter() method does not work on enu/enum columns, I suggest dropping the column and then creating it again the way you like. For example, if you want to change the enum text you can do something like this:
export async function up(knex: Knex): Promise<void> {
await knex.schema.alterTable('<table name>', (table) => {
table.dropColumn('column name');
});
await knex.schema.alterTable('<table name>', (table) => {
table.enu('column name', ['old enum 1', 'old enum 2'])
});
}
export async function down(knex: Knex): Promise<void> {
await knex.schema.alterTable('<table name>', (table) => {
table.dropColumn('column name');
});
await knex.schema.alterTable('<table name>', (table) => {
table.enu('column name', ['new enum 1', 'new enum2'])
});
}
Sources
This article follows the attribution requirements of Stack Overflow and is licensed under CC BY-SA 3.0.
Source: Stack Overflow
| Solution | Source |
|---|---|
| Solution 1 | Mikael Lepistö |
| Solution 2 | Kwame Opare Asiedu |
| Solution 3 | Eric Aya |
| Solution 4 |
