[英]How to do knex.js migrations?
我仍然不確定如何使用knex進行遷移。 這是我到目前為止所擁有的。 它的工作原理上up
,但down
給我FK約束錯誤,即使FOREIGN_KEY_CHECKS = 0。
exports.up = function(knex, Promise) {
return Promise.all([
knex.raw('SET foreign_key_checks = 0;'),
/* CREATE Member table */
knex.schema.createTable('Member', function (table) {
table.bigIncrements('id').primary().unsigned();
table.string('email',50);
table.string('password');
/* CREATE FKS */
table.bigInteger('ReferralId').unsigned().index();
table.bigInteger('AddressId').unsigned().index().inTable('Address').references('id');
}),
/* CREATE Address table */
knex.schema.createTable('Address', function (table) {
table.bigIncrements('id').primary().unsigned();
table.index(['city','state','zip']);
table.string('city',50).notNullable();
table.string('state',2).notNullable();
table.integer('zip',5).unsigned().notNullable();
}),
knex.raw('SET foreign_key_checks = 1;')
]);
};
exports.down = function(knex, Promise) {
return Promise.all([
knex.raw('SET foreign_key_checks = 0;'),
knex.schema.dropTable('Address'),
knex.schema.dropTable('Member'),
knex.raw('SET foreign_key_checks = 1;')
]);
};
jedd.ahyoung是正確的。 您不需要將連接池限制為1。只需要鏈接您的Promise,這樣它們就不會並行運行。
例如:
exports.up = function(knex, Promise) {
return removeForeignKeyChecks()
.then(createMemberTable)
.then(createAddressTable)
.then(addForeignKeyChecks);
function removeForeignKeyChecks() {
return knex.raw('SET foreign_key_checks = 0;');
}
function addForeignKeyChecks() {
return knex.raw('SET foreign_key_checks = 1;');
}
function createMemberTable() {
return knex.schema.createTable('Member', function (table) {
table.bigIncrements('id').primary().unsigned();
table.string('email',50);
table.string('password');
/* CREATE FKS */
table.bigInteger('ReferralId').unsigned().index();
table.bigInteger('AddressId').unsigned().index().inTable('Address').references('id');
});
}
function createAddressTable() {
return knex.schema.createTable('Address', function (table) {
table.bigIncrements('id').primary().unsigned();
table.index(['city','state','zip']);
table.string('city',50).notNullable();
table.string('state',2).notNullable();
table.integer('zip',5).unsigned().notNullable();
});
}
};
另外,我可能會丟失一些東西,但看起來如果您在成員表之前創建地址表,則無需刪除然后再恢復外鍵檢查。
最終代碼如下所示:
exports.up = function(knex, Promise) {
return createAddressTable()
.then(createMemberTable);
function createMemberTable() {
return knex.schema.createTable('Member', function (table) {
table.bigIncrements('id').primary().unsigned();
table.string('email',50);
table.string('password');
/* CREATE FKS */
table.bigInteger('ReferralId').unsigned().index();
table.bigInteger('AddressId').unsigned().index().inTable('Address').references('id');
});
}
function createAddressTable() {
return knex.schema.createTable('Address', function (table) {
table.bigIncrements('id').primary().unsigned();
table.index(['city','state','zip']);
table.string('city',50).notNullable();
table.string('state',2).notNullable();
table.integer('zip',5).unsigned().notNullable();
});
}
};
找出它是由於連接池不起作用。 它將使用不同的連接來運行每個遷移任務,這將導致外鍵檢查設置不正確。 設置
pool:{
max:1
}
在遷移配置文件中修復了此問題。
我通過交易解決了這個問題
transation.js
module.exports = function transaction(fn) {
return function _transaction(knex, Promise) {
return knex.transaction(function(trx) {
return trx
.raw('SET foreign_key_checks = 0;')
.then(function() {
return fn(trx, Promise);
})
.finally(function() {
return trx.raw('SET foreign_key_checks = 1;');
});
});
};
}
遷移檔案
var transaction = require('../transaction');
function up(trx, Promise) {
return trx.schema
.createTable('contract', function(table) {
table.boolean('active').notNullable();
table.integer('defaultPriority').unsigned().references('priority.id');
table.integer('defaultIssueStatus').unsigned().references('issueStatus.id');
table.integer('owner').notNullable().unsigned().references('user.id');
})
.createTable('user', function (table) {
table.increments('id').primary();
table.datetime('createdAt');
table.datetime('updatedAt');
table.string('phoneNumber').notNullable().unique();
table.string('password').notNullable();
table.string('name').notNullable().unique();
table.string('email');
table.string('status');
table.string('roles').defaultTo('user');
table.integer('contract').unsigned().references('contract.id');
});
}
function down(trx, Promise) {
return trx.schema
.dropTable('contract')
.dropTable('user');
}
exports.up = transaction(up);
exports.down = transaction(down);
inTable()
應該放在references()
:
inTablecolumn.inTable(table)
設置在調用
column.references
之后外鍵列所在的table
。
文件資料 。
我以為我會對此進行更新,因為Javascript中有一些新增功能,使此操作變得相當容易。 knex
仍然要求您返回一個Promise
但在該Promise
您可以做很多事情來清理與表的創建/修改有關的代碼。 我的首選是結合使用async
/ await
和Promise.all
。
exports.up = function(knex, Promise) {
return new Promise(async (resolve, reject) => {
try {
await Promise.all([
knex.schema.createTable('videos', table => {
table.increments('id');
table.string('title');
table.string('director');
table.json('meta');
}),
knex.schema.createTable('books', table => {
table.increments('id');
table.string('title');
table.string('author');
table.json('meta');
})
]);
console.log('Tables created successfully');
resolve();
} catch(error) {
reject(error);
}
})
}
如果您希望單獨創建每個表,那么我只需要使用async
/ await
。
exports.up = function(knex, Promise) {
return new Promise(async (resolve, reject) => {
try {
await knex.schema.createTable('videos', table => {
table.increments('id');
table.string('title');
table.string('director');
table.json('meta');
});
console.log('videos table created successfully!');
await knex.schema.createTable('books', table => {
table.increments('id');
table.string('title');
table.string('author');
table.json('meta');
});
console.log('books table created successfully!');
resolve();
} catch(error){
reject(error);
}
})
}
這使事情變得更加干凈,不需要您菊環串接then
綁定或包裝在周圍的函數中。 您只需要await
每個表的創建來解決,然后再解決它們封裝Promise
! 是的async
/ await
!
您也可以遵循這種模式在down
遷移中刪除表。 只需將knex.schema.createTable
語句替換為knex.schema.createTable
語句knex.schema.dropTable
。
從主觀上講,這是最干凈的方法,建議您在遷移文件中添加以下內容:
exports.up = function (knex) {
return Promise.all([
knex.schema.createTable('users', function (table) {
table.increments('id')
table.string('username').notNullable()
table.string('password').notNullable()
table.string('service').notNullable()
table.text('cookies')
table.enu('status', ['active', 'disabled', 'need_login', 'failed']).defaultTo('need_login').notNullable()
table.datetime('last_checked')
table.timestamps()
}),
knex.schema.createTable('products', function (table) {
table.increments()
table.integer('user_id').unsigned().notNullable()
table.string('title')
table.decimal('price', 8, 2)
table.text('notes')
table.enu('status', ['to_publish', 'published', 'hidden', 'not_found']).notNullable()
table.timestamps()
table.foreign('user_id').references('id').inTable('users')
}),
knex.schema.createTable('messages', function (table) {
table.increments()
table.integer('user_id').unsigned().notNullable()
table.integer('product_id').unsigned().notNullable()
table.boolean('incoming')
table.boolean('unread')
table.text('text')
table.timestamps()
table.foreign('user_id').references('id').inTable('users')
table.foreign('product_id').references('id').inTable('products')
})
])
}
exports.down = function (knex) {
return Promise.all([
knex.schema.dropTable('messages'),
knex.schema.dropTable('products'),
knex.schema.dropTable('users')
])
}
聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.