8
0
mirror of https://github.com/FirebirdSQL/firebird.git synced 2025-01-29 06:43:03 +01:00
firebird-mirror/src/jrd/trace/TraceService.cpp
2009-07-06 07:29:52 +00:00

350 lines
7.7 KiB
C++

/*
* MODULE: TraceService.cpp
* DESCRIPTION:
*
* The contents of this file are subject to the Initial
* Developer's Public License Version 1.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.ibphoenix.com/main.nfs?a=ibphoenix&page=ibp_idpl.
*
* Software distributed under the License is distributed AS IS,
* WITHOUT WARRANTY OF ANY KIND, either express or implied.
* See the License for the specific language governing rights
* and limitations under the License.
*
* The Original Code was created by Khorsun Vladyslav
* for the Firebird Open Source RDBMS project.
*
* Copyright (c) 2008 Khorsun Vladyslav <hvlad@users.sourceforge.net>
* and all contributors signed below.
*
* All Rights Reserved.
* Contributor(s): ______________________________________.
*
*/
#include "firebird.h"
#include "consts_pub.h"
#include "fb_exception.h"
#include "iberror.h"
#include "../../jrd/ibase.h"
#include "../../common/classes/fb_string.h"
#include "../../common/classes/timestamp.h"
#include "../../common/config/config.h"
#include "../../common/StatusArg.h"
#include "../../common/thd.h"
#include "../../jrd/svc.h"
#include "../../jrd/os/guid.h"
#include "../../jrd/trace/TraceLog.h"
#include "../../jrd/trace/TraceManager.h"
#include "../../jrd/trace/TraceService.h"
using namespace Firebird;
using namespace Jrd;
class TraceSvcJrd : public TraceSvcIntf
{
public:
explicit TraceSvcJrd(Service& svc) :
m_svc(svc),
m_admin(false),
m_chg_number(0)
{};
virtual ~TraceSvcJrd() {};
virtual void setAttachInfo(const string& service_name, const string& user,
const string& pwd, bool isAdmin);
virtual void startSession(TraceSession& session, bool interactive);
virtual void stopSession(ULONG id);
virtual void setActive(ULONG id, bool active);
virtual void listSessions();
private:
void readSession(TraceSession& session);
bool changeFlags(ULONG id, int setFlags, int clearFlags);
bool checkAliveAndFlags(ULONG sesId, int& flags);
Service& m_svc;
string m_user;
bool m_admin;
ULONG m_chg_number;
};
void TraceSvcJrd::setAttachInfo(const string& /*service_name*/, const string& user,
const string& /*pwd*/, bool isAdmin)
{
m_user = user;
m_admin = isAdmin || (m_user == SYSDBA_USER_NAME);
}
void TraceSvcJrd::startSession(TraceSession& session, bool interactive)
{
ConfigStorage* storage = TraceManager::getStorage();
{ // scope
StorageGuard guard(storage);
session.ses_user = m_user;
session.ses_flags = trs_active;
if (m_admin) {
session.ses_flags |= trs_admin;
}
if (interactive)
{
FB_GUID guid;
GenerateGuid(&guid);
char* buff = session.ses_logfile.getBuffer(GUID_BUFF_SIZE);
GuidToString(buff, &guid);
session.ses_logfile.insert(0, "fb_trace.");
}
storage->addSession(session);
m_chg_number = storage->getChangeNumber();
}
m_svc.started();
if (interactive)
{
readSession(session);
{
StorageGuard guard(storage);
storage->removeSession(session.ses_id);
}
}
else {
m_svc.printf("Trace session ID %ld started\n", session.ses_id);
}
}
void TraceSvcJrd::stopSession(ULONG id)
{
m_svc.started();
ConfigStorage* storage = TraceManager::getStorage();
StorageGuard guard(storage);
storage->restart();
TraceSession session(*getDefaultMemoryPool());
while (storage->getNextSession(session))
{
if (id != session.ses_id)
continue;
if (m_admin || m_user == session.ses_user)
{
storage->removeSession(id);
m_svc.printf("Trace session ID %ld stopped\n", id);
}
else
m_svc.printf("No permissions to stop other user trace session\n");
return;
}
m_svc.printf("Trace session ID %d not found\n", id);
}
void TraceSvcJrd::setActive(ULONG id, bool active)
{
if (active)
{
if (changeFlags(id, trs_active, 0)) {
m_svc.printf("Trace session ID %ld resumed\n", id);
}
}
else
{
if (changeFlags(id, 0, trs_active)) {
m_svc.printf("Trace session ID %ld paused\n", id);
}
}
}
bool TraceSvcJrd::changeFlags(ULONG id, int setFlags, int clearFlags)
{
ConfigStorage* storage = TraceManager::getStorage();
StorageGuard guard(storage);
storage->restart();
TraceSession session(*getDefaultMemoryPool());
while (storage->getNextSession(session))
{
if (id != session.ses_id)
continue;
if (m_admin || m_user == session.ses_user)
{
const int saveFlags = session.ses_flags;
session.ses_flags |= setFlags;
session.ses_flags &= ~clearFlags;
if (saveFlags != session.ses_flags) {
storage->updateSession(session);
}
return true;
}
m_svc.printf("No permissions to change other user trace session\n");
return false;
}
m_svc.printf("Trace session ID %d not found\n", id);
return false;
}
void TraceSvcJrd::listSessions()
{
m_svc.started();
ConfigStorage* storage = TraceManager::getStorage();
StorageGuard guard(storage);
storage->restart();
TraceSession session(*getDefaultMemoryPool());
while (storage->getNextSession(session))
{
if (m_admin || m_user == session.ses_user)
{
m_svc.printf("\nSession ID: %d\n", session.ses_id);
if (!session.ses_name.empty()) {
m_svc.printf(" name: %s\n", session.ses_name.c_str());
}
m_svc.printf(" user: %s\n", session.ses_user.c_str());
struct tm* t = localtime(&session.ses_start);
m_svc.printf(" date: %04d-%02d-%02d %02d:%02d:%02d\n",
t->tm_year + 1900, t->tm_mon + 1, t->tm_mday,
t->tm_hour, t->tm_min, t->tm_sec);
string flags;
if (session.ses_flags & trs_active) {
flags = "active";
}
else {
flags = "suspend";
}
if (session.ses_flags & trs_admin) {
flags += ", admin";
}
if (session.ses_flags & trs_system) {
flags += ", system";
}
if (session.ses_logfile.empty()) {
flags += ", audit";
}
else {
flags += ", trace";
}
if (session.ses_flags & trs_log_full) {
flags += ", log full";
}
m_svc.printf(" flags: %s\n", flags.c_str());
}
}
}
void TraceSvcJrd::readSession(TraceSession& session)
{
const size_t maxLogSize = Config::getMaxUserTraceLogSize(); // in MB
if (session.ses_logfile.empty())
{
m_svc.printf("Can't open trace data log file");
return;
}
MemoryPool& pool = *getDefaultMemoryPool();
AutoPtr<TraceLog> log(FB_NEW(pool) TraceLog(pool, session.ses_logfile, true));
UCHAR buff[1024];
int flags = session.ses_flags;
while (!m_svc.finished() && checkAliveAndFlags(session.ses_id, flags))
{
const size_t len = log->read(buff, sizeof(buff));
if (!len)
{
if (!checkAliveAndFlags(session.ses_id, flags))
break;
THD_sleep(250);
}
else
{
m_svc.putBytes(buff, len);
const bool logFull = (flags & trs_log_full);
if (logFull && log->getApproxLogSize() <= maxLogSize)
{
// resume session
changeFlags(session.ses_id, 0, trs_log_full);
}
}
}
}
bool TraceSvcJrd::checkAliveAndFlags(ULONG sesId, int& flags)
{
ConfigStorage* storage = TraceManager::getStorage();
bool alive = (m_chg_number == storage->getChangeNumber());
if (!alive)
{
// look if our session still alive
StorageGuard guard(storage);
TraceSession readSession(*getDefaultMemoryPool());
storage->restart();
while (storage->getNextSession(readSession))
{
if (readSession.ses_id == sesId)
{
alive = true;
flags = readSession.ses_flags;
break;
}
}
m_chg_number = storage->getChangeNumber();
}
return alive;
}
// service entrypoint
THREAD_ENTRY_DECLARE TRACE_main(THREAD_ENTRY_PARAM arg)
{
Service* svc = (Service*) arg;
int exit_code = FB_SUCCESS;
TraceSvcJrd traceSvc(*svc);
try
{
fbtrace(svc, &traceSvc);
}
catch (const Exception& e)
{
e.stuff_exception(svc->getStatus());
exit_code = FB_FAILURE;
}
svc->started();
svc->finish();
return (THREAD_ENTRY_RETURN)(IPTR) exit_code;
}