2012-06-07 22:55:06 +04:00
|
|
|
// Copyright 2012, Google Inc. All rights reserved.
|
|
|
|
// Use of this source code is governed by a BSD-style
|
|
|
|
// license that can be found in the LICENSE file.
|
2012-02-25 11:30:03 +04:00
|
|
|
|
|
|
|
package tabletserver
|
|
|
|
|
|
|
|
import (
|
2012-03-12 04:53:40 +04:00
|
|
|
"code.google.com/p/vitess/go/relog"
|
2012-02-25 11:30:03 +04:00
|
|
|
"fmt"
|
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
FAIL = iota
|
|
|
|
RETRY
|
|
|
|
FATAL
|
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
DUPLICATE_KEY = 1062 // MySQL error number
|
|
|
|
)
|
|
|
|
|
|
|
|
type TabletError struct {
|
|
|
|
ErrorType int
|
|
|
|
Message string
|
|
|
|
SqlError int
|
|
|
|
}
|
|
|
|
|
|
|
|
// This is how go-mysql exports its error number
|
|
|
|
type hasNumber interface {
|
|
|
|
Number() int
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewTabletError(errorType int, format string, args ...interface{}) *TabletError {
|
|
|
|
return &TabletError{errorType, fmt.Sprintf(format, args...), 0}
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewTabletErrorSql(errorType int, err error) *TabletError {
|
|
|
|
self := NewTabletError(errorType, "%s", err)
|
|
|
|
if sqlErr, ok := err.(hasNumber); ok {
|
|
|
|
self.SqlError = sqlErr.Number()
|
|
|
|
}
|
|
|
|
return self
|
|
|
|
}
|
|
|
|
|
|
|
|
func (self *TabletError) Error() string {
|
|
|
|
format := "error: %s"
|
|
|
|
switch self.ErrorType {
|
|
|
|
case RETRY:
|
|
|
|
format = "retry: %s"
|
|
|
|
case FATAL:
|
|
|
|
format = "fatal: %s"
|
|
|
|
}
|
|
|
|
return fmt.Sprintf(format, self.Message)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (self *TabletError) RecordStats() {
|
|
|
|
switch self.ErrorType {
|
|
|
|
case RETRY:
|
|
|
|
errorStats.Add("Retry", 1)
|
|
|
|
case FATAL:
|
|
|
|
errorStats.Add("Fatal", 1)
|
|
|
|
default:
|
|
|
|
if self.SqlError == DUPLICATE_KEY {
|
|
|
|
errorStats.Add("DupKey", 1)
|
|
|
|
} else {
|
|
|
|
errorStats.Add("Fail", 1)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-11-20 04:24:03 +04:00
|
|
|
func handleError(err *error, logStats *sqlQueryStats) {
|
|
|
|
if logStats != nil {
|
|
|
|
logStats.Send()
|
|
|
|
}
|
2012-02-25 11:30:03 +04:00
|
|
|
if x := recover(); x != nil {
|
2012-12-20 05:48:02 +04:00
|
|
|
terr, ok := x.(*TabletError)
|
|
|
|
if !ok {
|
|
|
|
relog.Error("Uncaught panic:\n%v\n%s", x, relog.Stack(4))
|
|
|
|
*err = NewTabletError(FAIL, "%v: uncaught panic", x)
|
|
|
|
errorStats.Add("Panic", 1)
|
|
|
|
return
|
|
|
|
}
|
2012-02-25 11:30:03 +04:00
|
|
|
*err = terr
|
|
|
|
terr.RecordStats()
|
|
|
|
if terr.ErrorType == RETRY { // Retry errors are too spammy
|
|
|
|
return
|
|
|
|
}
|
|
|
|
relog.Error("%s", terr.Message)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func logError() {
|
|
|
|
if x := recover(); x != nil {
|
|
|
|
relog.Error("%s", x.(*TabletError).Message)
|
|
|
|
}
|
|
|
|
}
|