2015-08-24 20:25:01 +02:00
// Go MySQL Driver - A MySQL-Driver for Go's database/sql package
//
// Copyright 2013 The Go-MySQL-Driver Authors. All rights reserved.
//
// This Source Code Form is subject to the terms of the Mozilla Public
// License, v. 2.0. If a copy of the MPL was not distributed with this file,
// You can obtain one at http://mozilla.org/MPL/2.0/.
package mysql
import (
"database/sql/driver"
"errors"
"fmt"
"io"
2017-05-02 21:58:54 +02:00
"log"
"os"
2015-08-24 20:25:01 +02:00
)
2017-05-02 21:58:54 +02:00
// Various errors the driver might return. Can change between driver versions.
2015-08-24 20:25:01 +02:00
var (
2017-05-02 21:58:54 +02:00
ErrInvalidConn = errors . New ( "invalid connection" )
ErrMalformPkt = errors . New ( "malformed packet" )
ErrNoTLS = errors . New ( "TLS requested but server does not support TLS" )
ErrCleartextPassword = errors . New ( "this user requires clear text authentication. If you still want to use it, please add 'allowCleartextPasswords=1' to your DSN" )
ErrNativePassword = errors . New ( "this user requires mysql native password authentication." )
ErrOldPassword = errors . New ( "this user requires old password authentication. If you still want to use it, please add 'allowOldPasswords=1' to your DSN. See also https://github.com/go-sql-driver/mysql/wiki/old_passwords" )
ErrUnknownPlugin = errors . New ( "this authentication plugin is not supported" )
ErrOldProtocol = errors . New ( "MySQL server does not support required protocol 41+" )
ErrPktSync = errors . New ( "commands out of sync. You can't run this command now" )
ErrPktSyncMul = errors . New ( "commands out of sync. Did you run multiple statements at once?" )
ErrPktTooLarge = errors . New ( "packet for query is too large. Try adjusting the 'max_allowed_packet' variable on the server" )
ErrBusyBuffer = errors . New ( "busy buffer" )
2015-08-24 20:25:01 +02:00
)
2017-05-02 21:58:54 +02:00
var errLog = Logger ( log . New ( os . Stderr , "[mysql] " , log . Ldate | log . Ltime | log . Lshortfile ) )
// Logger is used to log critical error messages.
type Logger interface {
Print ( v ... interface { } )
}
// SetLogger is used to set the logger for critical errors.
// The initial logger is os.Stderr.
func SetLogger ( logger Logger ) error {
if logger == nil {
return errors . New ( "logger is nil" )
}
errLog = logger
return nil
}
// MySQLError is an error type which represents a single MySQL error
2015-08-24 20:25:01 +02:00
type MySQLError struct {
Number uint16
Message string
}
func ( me * MySQLError ) Error ( ) string {
return fmt . Sprintf ( "Error %d: %s" , me . Number , me . Message )
}
2017-05-02 21:58:54 +02:00
// MySQLWarnings is an error type which represents a group of one or more MySQL
// warnings
type MySQLWarnings [ ] MySQLWarning
2015-08-24 20:25:01 +02:00
func ( mws MySQLWarnings ) Error ( ) string {
var msg string
for i , warning := range mws {
if i > 0 {
msg += "\r\n"
}
2017-05-02 21:58:54 +02:00
msg += fmt . Sprintf (
"%s %s: %s" ,
warning . Level ,
warning . Code ,
warning . Message ,
)
2015-08-24 20:25:01 +02:00
}
return msg
}
2017-05-02 21:58:54 +02:00
// MySQLWarning is an error type which represents a single MySQL warning.
// Warnings are returned in groups only. See MySQLWarnings
type MySQLWarning struct {
2015-08-24 20:25:01 +02:00
Level string
Code string
Message string
}
func ( mc * mysqlConn ) getWarnings ( ) ( err error ) {
2017-05-02 21:58:54 +02:00
rows , err := mc . Query ( "SHOW WARNINGS" , nil )
2015-08-24 20:25:01 +02:00
if err != nil {
return
}
var warnings = MySQLWarnings { }
var values = make ( [ ] driver . Value , 3 )
for {
err = rows . Next ( values )
switch err {
case nil :
2017-05-02 21:58:54 +02:00
warning := MySQLWarning { }
2015-08-24 20:25:01 +02:00
2017-05-02 21:58:54 +02:00
if raw , ok := values [ 0 ] . ( [ ] byte ) ; ok {
2015-08-24 20:25:01 +02:00
warning . Level = string ( raw )
} else {
warning . Level = fmt . Sprintf ( "%s" , values [ 0 ] )
}
2017-05-02 21:58:54 +02:00
if raw , ok := values [ 1 ] . ( [ ] byte ) ; ok {
2015-08-24 20:25:01 +02:00
warning . Code = string ( raw )
} else {
warning . Code = fmt . Sprintf ( "%s" , values [ 1 ] )
}
2017-05-02 21:58:54 +02:00
if raw , ok := values [ 2 ] . ( [ ] byte ) ; ok {
2015-08-24 20:25:01 +02:00
warning . Message = string ( raw )
} else {
warning . Message = fmt . Sprintf ( "%s" , values [ 0 ] )
}
warnings = append ( warnings , warning )
case io . EOF :
return warnings
default :
rows . Close ( )
return
}
}
}