2009-02-05 17:51:51 +01:00
|
|
|
// lasterror.h
|
|
|
|
|
2009-10-27 20:58:27 +01:00
|
|
|
/* Copyright 2009 10gen Inc.
|
|
|
|
*
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
*/
|
2009-01-14 23:17:24 +01:00
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include <boost/thread/tss.hpp>
|
2009-05-15 23:27:31 +02:00
|
|
|
#undef assert
|
|
|
|
#define assert xassert
|
2009-01-14 23:09:51 +01:00
|
|
|
|
|
|
|
namespace mongo {
|
2009-03-26 18:45:29 +01:00
|
|
|
class BSONObjBuilder;
|
2009-09-11 18:52:17 +02:00
|
|
|
class Message;
|
|
|
|
|
2009-01-15 16:17:11 +01:00
|
|
|
struct LastError {
|
2009-12-28 23:12:49 +01:00
|
|
|
int code;
|
2009-01-15 16:17:11 +01:00
|
|
|
string msg;
|
2009-04-06 23:57:16 +02:00
|
|
|
enum UpdatedExistingType { NotUpdate, True, False } updatedExisting;
|
2009-11-18 20:22:00 +01:00
|
|
|
/* todo: nObjects should be 64 bit */
|
2009-03-26 18:45:29 +01:00
|
|
|
int nObjects;
|
2009-01-15 16:17:11 +01:00
|
|
|
int nPrev;
|
2009-03-26 18:45:29 +01:00
|
|
|
bool valid;
|
2009-09-11 18:52:17 +02:00
|
|
|
bool overridenById;
|
2009-12-28 23:12:49 +01:00
|
|
|
void raiseError(int _code , const char *_msg) {
|
2009-03-26 18:45:29 +01:00
|
|
|
reset( true );
|
2009-12-28 23:12:49 +01:00
|
|
|
code = _code;
|
2009-01-15 16:17:11 +01:00
|
|
|
msg = _msg;
|
|
|
|
}
|
2009-04-10 16:30:57 +02:00
|
|
|
void recordUpdate( bool _updatedExisting, int nChanged ) {
|
2009-03-26 18:45:29 +01:00
|
|
|
reset( true );
|
2009-04-10 16:30:57 +02:00
|
|
|
nObjects = nChanged;
|
2009-03-26 18:45:29 +01:00
|
|
|
updatedExisting = _updatedExisting ? True : False;
|
2009-01-15 16:17:11 +01:00
|
|
|
}
|
2009-03-26 18:45:29 +01:00
|
|
|
void recordDelete( int nDeleted ) {
|
|
|
|
reset( true );
|
|
|
|
nObjects = nDeleted;
|
2009-01-15 16:17:11 +01:00
|
|
|
}
|
|
|
|
LastError() {
|
2009-09-11 18:52:17 +02:00
|
|
|
overridenById = false;
|
2009-03-26 18:45:29 +01:00
|
|
|
reset();
|
|
|
|
}
|
|
|
|
void reset( bool _valid = false ) {
|
2009-12-28 23:12:49 +01:00
|
|
|
code = 0;
|
2009-03-26 18:45:29 +01:00
|
|
|
msg.clear();
|
|
|
|
updatedExisting = NotUpdate;
|
|
|
|
nObjects = 0;
|
|
|
|
nPrev = 1;
|
|
|
|
valid = _valid;
|
2009-01-15 16:17:11 +01:00
|
|
|
}
|
2009-03-26 18:45:29 +01:00
|
|
|
void appendSelf( BSONObjBuilder &b );
|
|
|
|
static LastError noError;
|
2009-01-15 16:17:11 +01:00
|
|
|
};
|
|
|
|
|
2009-10-12 21:12:16 +02:00
|
|
|
extern class LastErrorHolder {
|
2009-09-11 17:56:57 +02:00
|
|
|
public:
|
2010-01-12 20:38:00 +01:00
|
|
|
LastErrorHolder() : _id( 0 ), _disabled() {}
|
2009-09-11 17:56:57 +02:00
|
|
|
|
2009-09-11 18:52:17 +02:00
|
|
|
LastError * get( bool create = false );
|
2009-10-12 21:12:16 +02:00
|
|
|
|
2009-09-11 17:56:57 +02:00
|
|
|
void reset( LastError * le );
|
|
|
|
|
|
|
|
/**
|
|
|
|
* id of 0 means should use thread local management
|
|
|
|
*/
|
|
|
|
void setID( int id );
|
2009-09-11 18:52:17 +02:00
|
|
|
int getID();
|
|
|
|
|
2009-09-11 17:56:57 +02:00
|
|
|
void remove( int id );
|
|
|
|
void release();
|
2009-09-11 18:52:17 +02:00
|
|
|
|
2009-10-12 21:12:16 +02:00
|
|
|
/** when db receives a message/request, call this */
|
|
|
|
void startRequest( Message& m , LastError * connectionOwned );
|
|
|
|
void startRequest( Message& m );
|
2010-01-12 20:38:00 +01:00
|
|
|
|
|
|
|
// used to disable lastError reporting while processing a killCursors message
|
|
|
|
// disable causes get() to return 0.
|
2010-01-13 00:33:29 +01:00
|
|
|
LastError *disableForCommand(); // only call once per command invocation!
|
2010-01-12 20:38:00 +01:00
|
|
|
void enable() { _disabled = false; }
|
|
|
|
bool disabled() { return _disabled; }
|
2009-09-11 17:56:57 +02:00
|
|
|
private:
|
2010-01-13 00:33:29 +01:00
|
|
|
void disable() { _disabled = true; }
|
2009-10-13 18:55:23 +02:00
|
|
|
ThreadLocalValue<int> _id;
|
2009-09-11 17:56:57 +02:00
|
|
|
boost::thread_specific_ptr<LastError> _tl;
|
|
|
|
|
2009-10-12 21:12:16 +02:00
|
|
|
struct Status {
|
|
|
|
time_t time;
|
|
|
|
LastError *lerr;
|
|
|
|
};
|
|
|
|
static boost::mutex _idsmutex;
|
2010-01-12 20:38:00 +01:00
|
|
|
map<int,Status> _ids;
|
|
|
|
bool _disabled;
|
2009-10-12 21:12:16 +02:00
|
|
|
} lastError;
|
2009-09-11 17:56:57 +02:00
|
|
|
|
2009-12-28 23:12:49 +01:00
|
|
|
inline void raiseError(int code , const char *msg) {
|
2009-01-15 16:17:11 +01:00
|
|
|
LastError *le = lastError.get();
|
|
|
|
if ( le == 0 ) {
|
2010-01-12 20:38:00 +01:00
|
|
|
if ( lastError.disabled() ) {
|
|
|
|
log() << "lastError disabled, can't report: " << msg << endl;
|
|
|
|
} else {
|
|
|
|
DEV log() << "warning: lastError==0 can't report:" << msg << '\n';
|
|
|
|
}
|
2009-01-15 16:17:11 +01:00
|
|
|
return;
|
|
|
|
}
|
2009-12-28 23:12:49 +01:00
|
|
|
le->raiseError(code, msg);
|
2009-01-14 23:17:24 +01:00
|
|
|
}
|
2009-03-26 18:45:29 +01:00
|
|
|
|
2009-04-10 16:30:57 +02:00
|
|
|
inline void recordUpdate( bool updatedExisting, int nChanged ) {
|
2009-03-26 18:45:29 +01:00
|
|
|
LastError *le = lastError.get();
|
|
|
|
if ( le )
|
2009-04-10 16:30:57 +02:00
|
|
|
le->recordUpdate( updatedExisting, nChanged );
|
2009-03-26 18:45:29 +01:00
|
|
|
}
|
2009-01-15 16:17:11 +01:00
|
|
|
|
2009-03-26 18:45:29 +01:00
|
|
|
inline void recordDelete( int nDeleted ) {
|
|
|
|
LastError *le = lastError.get();
|
|
|
|
if ( le )
|
|
|
|
le->recordDelete( nDeleted );
|
|
|
|
}
|
2009-11-18 20:54:37 +01:00
|
|
|
|
2009-01-15 16:17:11 +01:00
|
|
|
} // namespace mongo
|