MySQL/Débogage
Introduction
modifierComme vu précédemment dans la description de la base "mysql", il peut être utile d'activer les logs sur l'historique des requêtes lors du débogage d'une application qui utilise MySQL.
Gestion des exceptions
modifierEn MySQL, les anomalies du type "division par zéro" ne renvoient pas d'erreur mais NULL.
Toutefois il est possible de lever des exceptions lors des manipulations de table, par exemple pour éviter qu'une liste d'insertions s'arrête au milieu à cause d'une contrainte d'unicité. L'exemple ci-dessous fonctionne sur une table InnoDB (et pas MyISAM)[1] :
ALTER TABLE `MaTable1` ADD UNIQUE(`id`);
INSERT INTO MaTable1 (id) VALUES('1');
START TRANSACTION;
INSERT INTO MaTable1 (id) VALUES('2');
INSERT INTO MaTable1 (id) VALUES('3');
INSERT INTO MaTable1 (id) VALUES('1');
IF condition THEN
COMMIT;
ELSE
ROLLBACK;
END IF;
Ici une erreur surgit lors de la deuxième insertion d'un id=1. Selon une condition, on peut donc annuler les insertions de 2 et 3, ou bien les soumettre.
SET autocommit = 0;
S'il y a plusieurs COMMIT avant un ROLLBACK (par exemple dans une boucle), ce dernier n'annulera que les opérations consécutives au dernier COMMIT.
Erreurs MySQL
modifierA new statement was found, but no delimiter between it and the previous one
modifierAjouter un ";".
Sinon avec "DELIMITER" : sans espace avant (donc consécutivement au retour chariot).
Access denied for user mon_compte@mon_hôte
modifier- Vérifier le mot de passe du compte.
- Vérifier que l'hôte du compte n'a pas été changé par une règle réseau.
- Si c'est le cas, tester en shell :
sudo mysql -h mon_hôte1 -u mon_compte sudo mysql -h mon_hôte2 -u mon_compte
- Sinon, recréer le droit avec :
GRANT ALL PRIVILEGES ON *.* TO 'mon_compte'@'mon_hôte2' IDENTIFIED BY PASSWORD 'mon_mot_de_passe_hashé';
Base table or view not found: 1146 Table 'xxx' doesn't exist" is not defined, use one of: 100, 200, 250, 300, 400, 500, 550, 600
modifierUne table temporaire n'est pas accessible, elle a peut-être été créée dans un autre thread : il faut la déplacer dans le même fichier.
BIGINT UNSIGNED value is out of range
modifierSe produit sur les opérations sur les grands nombres. On peut alors manipuler des nombres plus petits, par exemple en convertissant les octets en gigas dans le select.
Can't create table 'xxx' (errno: 150)
modifierClé primaire invalide. Dans ce cas il vaut mieux séparer la création de la table puis l'ajout des contraintes en deux requêtes. Cela peut permettre par exemple de voir que les tables vers lesquelles pointent les contraintes doivent être créées avant ces dernières.
Can't connect to MySQL server
modifierChanger le paramètre "host".
Can't create federated table
modifierLe schéma de la table distante doit être le même que celui de la table locale.
Sinon, activer les logs sur le serveur distant pour voir le compte qui se connecte.
Can't create/write to file 'mon_dossier/mon_fichier.csv'
modifierLors d'un SELECT INTO OUTFILE, si le compte du SGBD utilisé a les droits d'exportation (GRANT FILE ON .* TO 'mon_compte'@'%';
), le problème est lié au compte MySQL de l'OS qui n'a pas accès au système de fichier. Comme généralement le compte "mysql" sur Linux n'a pas de shell, cela peut se confirmer ainsi :
su - -s /bin/bash mysql touch mon_dossier/mon_fichier.csv
Si les droits manquent :
sudo chown -R mysql mon_dossier
Cannot add foreign key constraint
modifierSurvient lors d'un "CREATE TABLE", et ce n'est pas lié à la valeur de foreign_key_checks.
En fait il ne faut pas créer de contrainte d'intégrité entre deux tables de moteur différent (ex : InnoDB vs MyISAM).
Cannot drop index 'UNIQ_XXX': needed in a foreign key constraint
modifierIl faut supprimer la clé étrangère avant l'index duquel elle dépend :
ALTER TABLE `maTable` DROP FOREIGN KEY `FK_XXX`;
ALTER TABLE `maTable` DROP INDEX `UNIQ_XXX`;
... check that it exists
modifierAjouter IF EXISTS
. Ex :
DROP DATABASE IF EXISTS base_2; DROP TABLE IF EXISTS table_2; DROP TRIGGER IF EXISTS trigger_2; DROP EVENT IF EXISTS event_2;
IF EXISTS
dans les ALTER TABLE
. Ex :
ALTER TABLE table_1 DROP IF EXISTS table_2_id; ALTER TABLE table_1 DROP FOREIGN KEY IF EXISTS table_1_ibfk_1;
Connection refused
modifierSi tous les utilisateurs de la base ont cette erreur, relancer le serveur MySQL.
Par contre, un utilisateur peut se loguer depuis PhpMyAdmin, ou en ligne de commande (mysql -u mon_user ma_base -p
), mais pas depuis un DSN, si le serveur a un replica, vérifier sa configuration[2].
Data too long for column
modifierSi les données avec des caractères non ASCII étaient censées tenir dans le champ, on peut passer en mode strict :
SET @@global.sql_mode= 'NO_AUTO_CREATE_USER,NO_ENGINE_SUBSTITUTION';
Toutefois cela ne préserve pas leur encodage Unicode. Le mieux est donc :
SET NAMES 'utf8';
Failed to open the referenced table
modifierUne clé étrangère tente d'être créée vers une table inexistante ou qui ne supporte pas les clés étrangères (ex : moteur MyISAM).
Foreign data src error
modifierLors du create server
, le compte renseigné n'existe pas.
Foreign key constraint is incorrectly formed
modifierUne clé étrangère n'est pas au même format que la colonne à laquelle elle fait référence. Par exemple si cette dernière n'existe pas ou si elle n'a pas le même type.
Got error 'invalid character range'
modifierL'utilisation de \-
est parfois proscrite en regex (ou il faut l'échapper autrement ?).
Host 'example.com' is not allowed to connect to this MySQL server
modifierDans le cas d'une connexion depuis un PC distant, le compte utilisé n'est pas autorisé. Il faut donc le configurer avec :
GRANT ALL PRIVILEGES ON *.* TO 'utilisateur'@'%' WITH GRANT OPTION;
au lieu ou en plus de :
GRANT ALL PRIVILEGES ON *.* TO 'utilisateur'@'localhost' WITH GRANT OPTION;
Illegal mix of collations
modifierTrois solutions :
ALTER TABLE
pour changer la structure d'au moins une des deux tables jointes, pour uniformiser leur collation.CAST(monChamp AS CHAR CHARACTER SET utf8)
.CONVERT(monChamp USING utf8)
.
Invalid use of group function
modifier- Dans le cas d'un
SELECT
, il conviendrait d'utiliserHAVING
au lieu deWHERE
pour modifier des enregistrements en fonction d'autres d'une sous-requête. - Pour un
UPDATE
ou unDELETE
, les champs comparés par unIN
ne sont peut-être pas du même type.
Integrity constraint violation: 1217 Cannot delete or update a parent row: a foreign key constraint fails
modifier- En cas d'ajout de contrainte, il y a au moins un enregistrement qui ne peut pas être join avec la table cible.
- En cas de suppression, il manque un "DROP FOREIGN KEY" avant un "DROP TABLE" ou un "DROP COLUMN". Pour les supprimer, l'utilisateur root n'a pas le droit de modifier directement directement la table "information_schema". Il faut donc exécuter les requêtes générées par la suivante :
SELECT concat('alter table ',table_schema,'.',table_name,' DROP FOREIGN KEY ',constraint_name,';')
FROM information_schema.table_constraints
WHERE constraint_type='FOREIGN KEY'
AND table_name='maTable';
La clé est trop longue. Longueur maximale: 1000
modifierDans le cas d'un alter table général ou de plusieurs champs, décomposer la requête pour traiter les champs un par un afin d'isoler celui qui provoque l'erreur. Ensuite on peut la résoudre en réduisant la taille du varchar (ex : de 255 à 50 si possible).
Lock wait timeout exceeded; try restarting transaction
modifierLe serveur SQL est surchargé : exécuter les instructions une par une. Sinon :
SET innodb_lock_wait_timeout = 120;
No such file or directory
modifierRemplacer "localhost" par 127.0.0.1.
Query execution was interrupted, max_execution_time exceeded
modifierSET GLOBAL MAX_EXECUTION_TIME = 1000; -- milisecondes
-- Dans MariaDB :
SET GLOBAL max_statement_time = 1; -- secondes
View ... references invalid table(s) or column(s) or function(s) or definer/invoker of view lack rights to use them
modifierTemporairement on peut recréer la view sans "SQL SECURITY INVOKER".
Specified key was too long; max key length is 3072 bytes
modifierSe produit lors d'un ALTER vers un grand type comme "text". Exemple :
ALTER TABLE ma_table
CHANGE `mon_champ2` `mon_champ2` text COLLATE 'utf8_general_ci' NULL AFTER `mon_champ1`;
Solution :
ALTER TABLE ma_table
CHANGE `mon_champ2` `mon_champ2` varchar(1000) COLLATE 'utf8_general_ci' NULL AFTER `mon_champ1`;
Syntax error or access violation
modifierUtiliser phpMyAdmin pour trouver l'erreur de syntaxe.
Si cela survient après avoir collé une exportation de base, utiliser plutôt l'importation d'un .sql qui gère mieux les caractères d'échappement.
This version of MySQL doesn't yet support 'LIMIT & IN/ALL/ANY/SOME subquery'
modifierRemplacer les "IN" par des jointures, ou la sous-requête par une deuxième dont le résultat est stockée dans une table temporaire.
Too many tables (ou timeouts systématiques)
modifierSi des requêtes sont systématiquement en timeout, c'est peut-être que la limite de 61 jointures a été atteinte. Pour pallier cela, il est possible d'en remplacer certaines par des SELECT[3].
Type d'énoncé non reconnu
modifierCertains mots clés ne sont reconnus que dans les procédures stockées, ou doivent être précédés d'un SELECT.
Unexpected end of JSON text in argument 1 to function 'json_extract'
modifierLever l'exception vide avec : if(x = "", null, json_extract(x, y))
.
Unknown storage engine 'federated' mariadb
modifierLancer "mysql" en shell puis taper :
install plugin federated soname 'ha_federated.so';
'utf8mb3' is deprecated and will be removed in a future release
modifierPas besoin de changer chaque champ varchar ou text, il suffit de convertir la table :
ALTER TABLE ma_table CONVERT TO CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci;
You can't specify target table '...' for update in FROM clause
modifierCela se produit quand on essaie de mettre à jour ou supprimer des lignes selon une sélection de ces mêmes lignes. En effet, il est impossible de mettre à jour une table pendant en même temps qu'elle une subit sous-requête. Par exemple, pour réinitialiser un mot de passe SPIP :
mysql> UPDATE spip_auteurs SET pass = (SELECT pass FROM spip_auteurs WHERE login='paul') where login='admin'; ERROR 1093 (HY000): You can't specify target table 'spip_auteurs' for update in FROM clause
- Passer par des
CREATE TEMPORARY TABLE
(voireDECLARE
si cela rentre dans une variable scalaire). - Sinon, il est possible de sélectionner les enregistrements à mettre à jour automatiquement en enveloppant la sous-requête dans une autre, grâce aux tables temporaires générées par les
FROM
[4].
You do not have the SUPER privilege and binary logging is enabled
modifierSET GLOBAL log_bin_trust_function_creators = 1;
phpMyAdmin
modifierLes résultats sont tronqués
modifierCliquer sous la requête à exécuter dans "Options supplémentaires", puis "Textes complets".
Erreur : fonctionnalités relationnelles désactivées !
modifierSe produit dans le concepteur de diagramme de phpMyAdmin, il faut l'activer dans config.inc.php.
Références
modifier- ↑ http://stackoverflow.com/questions/2950676/difference-between-set-autocommit-1-and-start-transaction-in-mysql-have-i-misse
- ↑ https://dba.stackexchange.com/questions/55304/connection-refused-when-trying-to-set-up-replication
- ↑ https://www.quora.com/How-do-I-override-a-max-limit-of-61-joins-in-a-MySQL-query
- ↑ (anglais) http://www.xaprb.com/blog/2006/06/23/how-to-select-from-an-update-target-in-mysql/