Are you stuck with the error message ‘SQL error 1071’ and looking for a solution? We can help you resolve this SQL error
This SQL error mainly occurs if the combined key is too long. So by adjusting the varchar value, we can resolve this error message.
Here at Bobcares, we have seen several such SQL-related errors as part of our Server Management Services for web hosts and online service providers.
Today, let us see why this SQL error message occurs and also take a look at how our Support Engineers resolve this error.
How we resolve ‘SQL error 1071’
Now let us take a look at how our Support Engineers resolve this SQL error message for our customers.
1. Recently, one of our customers came across the error message ‘#1071 – Specified key was too long; max key length is 767 bytes’ while running the below query.
CREATE TABLE wp_locations ( `id` INT(11) NOT NULL AUTO_INCREMENT, `place` VARCHAR(255) NOT NULL, `name` VARCHAR(255) NOT NULL, CONSTRAINT `place_name` UNIQUE (`city`, `name`) ) DEFAULT CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci;
Here it is clear that the combined key is too long. So we need to either make separate keys or reduce the column lengths.
Generally, MySQL always reserves the max amount for a UTF8 field which is 4 bytes so with 255 + 255 with the DEFAULT CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci; this query is over the 767 max key length limit.
So we suggested our customer reduce the single varchar length or not use a composite key.
So our customer resolved this error by reducing the varchar value to 128.
2. Another way to fix such an error is to add the below lines in /etc/my.conf.d directory named umb4-support.cnf
[mysqld] innodb_large_prefix=true innodb_file_format=barracuda innodb_file_per_table=true
After that, we can restart the SQL service.
[Need any further assistance with SQL-related errors? – We are here to help you.]
Conclusion
In short, this ‘SQL error 1071’ mainly occurs if the combined key is too long, and adjusting the varchar value must resolve this error. Today, we saw how our Support Engineers resolve this SQL error.
PREVENT YOUR SERVER FROM CRASHING!
Never again lose customers to poor server speed! Let us help you.
Our server experts will monitor & maintain your server 24/7 so that it remains lightning fast and secure.
GET STARTED
var google_conversion_label = «owonCMyG5nEQ0aD71QM»;
Ispirer Home Page
Ispirer SQLWays Product Page — Migration to MySQL
Request SQLWays
Symptoms
During the import to MySQL, when you create a key for a InnoDB table, the “ERROR 1071 (42000)” error arises. For example,
CREATE TABLE department ( id INT, name VARCHAR(1000) ); ALTER TABLE department ADD PRIMARY KEY (id, name); ERROR 1071 (42000) at line 8: Specified key was too long; max key length is 767 bytes
Cause
The maximum key length for the MySQL database for InnoDB Engine is 767 bytes. The key length includes the sum lengths of all the columns included in the key.
Solutions
There are several solutions for this issue:
-
Changing the size of some columns. For example, if VARCHAR(300) is used to store last names, you can reduce its size to 100 with minimal risk of data loss.
-
Specifying the partial column length for the character columns in the key. For example, for VARCHAR(1000) column, you can specify only 100 characters to be used in the key.
ALTER TABLE department ADD PRIMARY KEY (id, name(100));
In this example only 100 bytes of the NAME column participate in the primary key.
-
Using a different set of columns for the key.
Ispirer Home Page
Ispirer SQLWays Product Page — Migration to MySQL
Request SQLWays
When restoring a MySQL database, I got the error “Specified key was too long; max key length is 767 bytes”.
user@host [~]$ mysql new_db < wordpress—2021—12—19—b51f8a6.sql ERROR 1071 (42000) at line 745 in file: ‘wordpress-2021-12-19-b51f8a6.sql’: Specified key was too long; max key length is 767 bytes user@host [~]$ |
MySQL version 5.6 and older versions have a limit of 767 bytes prefix limit.
https://dev.mysql.com/doc/refman/5.6/en/create-index.html
Prefix support and lengths of prefixes (where supported) are storage engine dependent. For example, a prefix can be up to 767 bytes long for InnoDB tables or 3072 bytes if the innodb_large_prefix option is enabled. For MyISAM tables, the prefix length limit is 1000 bytes.
From the error message, the error was on line 745, when checking the SQL file, I found the following SQL statement.
CREATE TABLE `wpk4_gla_merchant_issues` ( `id` bigint(20) NOT NULL AUTO_INCREMENT, `product_id` bigint(20) NOT NULL, `issue` varchar(200) COLLATE utf8mb4_unicode_520_ci NOT NULL, `code` varchar(100) COLLATE utf8mb4_unicode_520_ci NOT NULL, `severity` varchar(20) COLLATE utf8mb4_unicode_520_ci NOT NULL DEFAULT ‘warning’, `product` varchar(100) COLLATE utf8mb4_unicode_520_ci NOT NULL, `action` varchar(100) COLLATE utf8mb4_unicode_520_ci NOT NULL, `action_url` varchar(100) COLLATE utf8mb4_unicode_520_ci NOT NULL, `applicable_countries` text COLLATE utf8mb4_unicode_520_ci NOT NULL, `source` varchar(10) COLLATE utf8mb4_unicode_520_ci NOT NULL DEFAULT ‘mc’, `type` varchar(10) COLLATE utf8mb4_unicode_520_ci NOT NULL DEFAULT ‘product’, `created_at` datetime NOT NULL, PRIMARY KEY (`id`), UNIQUE KEY `product_issue` (`product_id`,`issue`) ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_520_ci; |
To fix it, find
UNIQUE KEY `product_issue` (`product_id`,`issue`) |
Replace with
UNIQUE KEY `product_issue` (`product_id`,`issue`(191)) |
This will limit the length of the issue column to 191 chars. Each character takes 4 bytes to store in utf8mb4 character set. The limit 191 is found by trying with default length for issue field 200, then reducing it until the SQL gets created properly. The calculation is 191 * 4 + length of id column < 767
The correct solution is to use newer MySQL versions like MySQL 5.7, which allows length up to 3072 bytes.
Мы столкнулись с этой проблемой при попытке добавить индекс UNIQUE в поле VARCHAR (255), используя utf8mb4. Хотя проблема здесь уже хорошо описана, я хотел бы добавить некоторые практические советы о том, как мы это поняли и решили.
При использовании utf8mb4 символы считаются 4 байтами, тогда как под utf8 они могут составлять 3 байта. Базы данных InnoDB имеют ограничение на то, что индексы могут содержать только 767 байт. Поэтому при использовании utf8 вы можете сохранить 255 символов (767/3 = 255), но используя utf8mb4, вы можете хранить только 191 символ (767/4 = 191).
Вы абсолютно можете добавлять регулярные индексы для полей VARCHAR(255)
, используя utf8mb4, но случается, что размер индекса усекается с 191 символом автоматически — например unique_key
здесь:
Это прекрасно, потому что регулярные индексы просто используются для быстрого поиска MySQL через ваши данные. Не нужно индексировать все поле.
Итак, почему MySQL автоматически обрезает индекс для обычных индексов, но бросает явную ошибку при попытке сделать это для уникальных индексов? Ну, для того, чтобы MySQL смог выяснить, существует ли уже существующее или обновляемое значение, ему нужно фактически индексировать все значение, а не только его часть.
В конце дня, если вы хотите иметь уникальный индекс в поле, все содержимое поля должно вписываться в индекс. Для utf8mb4 это означает сокращение длины полей VARCHAR до 191 символа или меньше. Если вам не нужна utf8mb4 для этой таблицы или поля, вы можете вернуть ее обратно в utf8 и сохранить свои длины длины.
Странная ошибка, возникающая при выполнении первой миграции в Laravel 8, тянется ещё со времён Laravel 5.4. И её почему-то до сих пор не устранили. Связано ли это с тем, что программисты, работающие с Laravel, предпочитают исключительно MySQL и люто ненавидят MariaDB, или ещё по какой причине, но ошибка случается и поправить её на самом деле, не составляет труда.
Ошибка выглядит следующим образом. При выполнении команды:
php artisan migrate
Миграции начинают обрабатываться:
Migration table created successfully.
Migrating: 2014_10_12_000000_create_users_table
Но тут же вылетает сообщение об ошибке:
IlluminateDatabaseQueryException
SQLSTATE[42000]: Syntax error or access violation: 1071 Specified key was too long; max key length is 767 bytes (SQL: alter table `users` add unique `users_email_unique`(`email`))
at vendor/laravel/framework/src/Illuminate/Database/Connection.php:671
667▕ // If an exception occurs when attempting to run a query, we'll format the error
668▕ // message to include the bindings with SQL, which will make this exception a
669▕ // lot more helpful to the developer instead of just the database's errors.
670▕ catch (Exception $e) {
➜ 671▕ throw new QueryException(
672▕ $query, $this->prepareBindings($bindings), $e
673▕ );
674▕ }
675▕
+9 vendor frames
10 database/migrations/2014_10_12_000000_create_users_table.php:26
IlluminateSupportFacadesFacade::__callStatic("create")
+21 vendor frames
32 artisan:37
IlluminateFoundationConsoleKernel::handle(Object(SymfonyComponentConsoleInputArgvInput), Object(SymfonyComponentConsoleOutputConsoleOutput))
Решение проблемы с ошибкой php artisan migrate
— SQLSTATE[42000]: Syntax error or access violation: 1071 Specified key was too long; max key length is 767 bytes при применении миграции в Laravel
Для устранения этой ошибки нужно внести изменения в метод boot
класса AppServiceProvider
. Для этого открываем файл, находящийся по адресу:
/папка_проекта/app/Providers/AppServiceProvider.php
Изначально он имеет вид:
<?php
namespace AppProviders;
use IlluminateSupportServiceProvider;
class AppServiceProvider extends ServiceProvider
{
/**
* Register any application services.
*
* @return void
*/
public function register()
{
//
}
/**
* Bootstrap any application services.
*
* @return void
*/
public function boot()
{
//
}
}
В него нужно вписать 2 строчки (использование библиотеки фасадов use IlluminateSupportFacadesSchema;
и модификацию метода boot
: Schema::defaultStringLength(191);
) таким вот образом:
use IlluminateSupportFacadesSchema;
public function boot()
{
Schema::defaultStringLength(191);
}
Таким образом модифицированный файл с классом AppServiceProvider
будет иметь вид (добавляется 6-я и 26-я строки):
<?php
namespace AppProviders;
use IlluminateSupportServiceProvider;
use IlluminateSupportFacadesSchema;
class AppServiceProvider extends ServiceProvider
{
/**
* Register any application services.
*
* @return void
*/
public function register()
{
//
}
/**
* Bootstrap any application services.
*
* @return void
*/
public function boot()
{
Schema::defaultStringLength(191);
}
}
После сохранения этих изменений, команда php artisan migrate
работает без ошибок, все миграции применяются и проект на фреймворке Laravel оживает. =)
Важно(!)
Перед тем, как накатывать повторно миграции, удалите созданные ранее таблицы из базы данных.
Иначе будет новая ошибка!
По материалам:
laravel-news.com
Заберите ссылку на статью к себе, чтобы потом легко её найти!
Выберите, то, чем пользуетесь чаще всего: