2009-10-21 02:42:38 +02:00
|
|
|
/*
|
|
|
|
* 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 Adriano dos Santos Fernandes
|
|
|
|
* for the Firebird Open Source RDBMS project.
|
|
|
|
*
|
|
|
|
* Copyright (c) 2008 Adriano dos Santos Fernandes <adrianosf@uol.com.br>
|
|
|
|
* and all contributors signed below.
|
|
|
|
*
|
|
|
|
* All Rights Reserved.
|
|
|
|
* Contributor(s): ______________________________________.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef JRD_EXT_ENGINE_MANAGER_H
|
|
|
|
#define JRD_EXT_ENGINE_MANAGER_H
|
|
|
|
|
|
|
|
#include "FirebirdApi.h"
|
2011-06-02 17:57:08 +02:00
|
|
|
#include "firebird/ExternalEngine.h"
|
2009-10-21 02:42:38 +02:00
|
|
|
#include "../common/classes/array.h"
|
|
|
|
#include "../common/classes/fb_string.h"
|
|
|
|
#include "../common/classes/GenericMap.h"
|
|
|
|
#include "../common/classes/MetaName.h"
|
2010-11-21 04:47:29 +01:00
|
|
|
#include "../common/classes/NestConst.h"
|
2009-10-21 02:42:38 +02:00
|
|
|
#include "../common/classes/auto.h"
|
|
|
|
#include "../common/classes/rwlock.h"
|
2011-05-19 18:24:46 +02:00
|
|
|
#include "../common/classes/ImplementHelper.h"
|
2011-06-01 03:44:54 +02:00
|
|
|
#include "../common/StatementMetadata.h"
|
2009-10-21 02:42:38 +02:00
|
|
|
|
|
|
|
struct dsc;
|
|
|
|
|
|
|
|
namespace Jrd {
|
|
|
|
|
|
|
|
class thread_db;
|
|
|
|
class jrd_prc;
|
2011-05-07 21:52:44 +02:00
|
|
|
class jrd_tra;
|
2009-10-21 02:42:38 +02:00
|
|
|
class Attachment;
|
|
|
|
class Database;
|
|
|
|
class Format;
|
|
|
|
class Trigger;
|
2009-12-21 18:43:01 +01:00
|
|
|
class Function;
|
2010-11-21 04:47:29 +01:00
|
|
|
class ValueExprNode;
|
2009-10-21 02:42:38 +02:00
|
|
|
struct impure_value;
|
|
|
|
struct record_param;
|
|
|
|
|
|
|
|
|
2013-01-29 02:21:32 +01:00
|
|
|
class RoutineMessage :
|
|
|
|
public Firebird::VersionedIface<Firebird::IRoutineMessage, FB_ROUTINE_MESSAGE_VERSION>,
|
|
|
|
public Firebird::PermanentStorage
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
explicit RoutineMessage(MemoryPool& pool)
|
|
|
|
: PermanentStorage(pool),
|
|
|
|
blr(pool),
|
|
|
|
bufferLength(0)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void FB_CARG set(const unsigned char* aBlr, unsigned aBlrLength, unsigned aBufferLength)
|
|
|
|
{
|
|
|
|
blr.assign(aBlr, aBlrLength);
|
|
|
|
bufferLength = aBufferLength;
|
|
|
|
}
|
|
|
|
|
|
|
|
public:
|
|
|
|
Firebird::Array<UCHAR> blr;
|
|
|
|
unsigned bufferLength;
|
|
|
|
};
|
|
|
|
|
|
|
|
class TriggerMessage :
|
|
|
|
public Firebird::VersionedIface<Firebird::ITriggerMessage, FB_TRIGGER_MESSAGE_VERSION>,
|
|
|
|
public Firebird::PermanentStorage
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
explicit TriggerMessage(MemoryPool& pool)
|
|
|
|
: PermanentStorage(pool),
|
|
|
|
names(pool),
|
|
|
|
blr(pool),
|
|
|
|
bufferLength(0)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void FB_CARG set(const unsigned char* aBlr, unsigned aBlrLength, unsigned aBufferLength,
|
|
|
|
const char** aNames, unsigned aCount)
|
|
|
|
{
|
|
|
|
blr.assign(aBlr, aBlrLength);
|
|
|
|
bufferLength = aBufferLength;
|
|
|
|
|
|
|
|
for (unsigned i = 0; i < aCount; ++i)
|
|
|
|
names.add(aNames[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
public:
|
|
|
|
Firebird::ObjectsArray<Firebird::string> names;
|
|
|
|
Firebird::Array<UCHAR> blr;
|
|
|
|
unsigned bufferLength;
|
|
|
|
};
|
|
|
|
|
2009-10-21 02:42:38 +02:00
|
|
|
class ExtEngineManager : public Firebird::PermanentStorage
|
|
|
|
{
|
|
|
|
private:
|
|
|
|
class AttachmentImpl;
|
|
|
|
template <typename T> class ContextManager;
|
|
|
|
class TransactionImpl;
|
|
|
|
|
2013-01-29 02:21:32 +01:00
|
|
|
class RoutineMetadata :
|
|
|
|
public Firebird::VersionedIface<Firebird::IRoutineMetadata, FB_ROUTINE_METADATA_VERSION>,
|
|
|
|
public Firebird::PermanentStorage
|
2011-05-15 17:37:11 +02:00
|
|
|
{
|
|
|
|
public:
|
2011-06-24 08:34:16 +02:00
|
|
|
explicit RoutineMetadata(MemoryPool& pool)
|
2011-05-15 17:37:11 +02:00
|
|
|
: PermanentStorage(pool),
|
|
|
|
package(pool),
|
|
|
|
name(pool),
|
|
|
|
entryPoint(pool),
|
|
|
|
body(pool),
|
|
|
|
triggerTable(pool),
|
|
|
|
triggerType(Firebird::ExternalTrigger::Type(0))
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual const char* FB_CARG getPackage(Firebird::IStatus* /*status*/) const
|
|
|
|
{
|
|
|
|
return package.nullStr();
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual const char* FB_CARG getName(Firebird::IStatus* /*status*/) const
|
|
|
|
{
|
|
|
|
return name.c_str();
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual const char* FB_CARG getEntryPoint(Firebird::IStatus* /*status*/) const
|
|
|
|
{
|
|
|
|
return entryPoint.c_str();
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual const char* FB_CARG getBody(Firebird::IStatus* /*status*/) const
|
|
|
|
{
|
|
|
|
return body.c_str();
|
|
|
|
}
|
|
|
|
|
2013-02-17 13:08:53 +01:00
|
|
|
virtual Firebird::IMessageMetadata* FB_CARG getInputMetadata(
|
2011-06-01 03:44:54 +02:00
|
|
|
Firebird::IStatus* /*status*/) const
|
|
|
|
{
|
2013-02-17 13:08:53 +01:00
|
|
|
return getMetadata(inputParameters);
|
2011-06-01 03:44:54 +02:00
|
|
|
}
|
|
|
|
|
2013-02-17 13:08:53 +01:00
|
|
|
virtual Firebird::IMessageMetadata* FB_CARG getOutputMetadata(
|
2011-06-01 03:44:54 +02:00
|
|
|
Firebird::IStatus* /*status*/) const
|
|
|
|
{
|
2013-02-17 13:08:53 +01:00
|
|
|
return getMetadata(outputParameters);
|
2011-06-01 03:44:54 +02:00
|
|
|
}
|
|
|
|
|
2013-02-17 13:08:53 +01:00
|
|
|
virtual Firebird::IMessageMetadata* FB_CARG getTriggerFields(
|
2011-06-01 03:44:54 +02:00
|
|
|
Firebird::IStatus* /*status*/) const
|
|
|
|
{
|
2013-02-17 13:08:53 +01:00
|
|
|
return getMetadata(triggerFields);
|
2011-06-01 03:44:54 +02:00
|
|
|
}
|
|
|
|
|
2011-05-15 17:37:11 +02:00
|
|
|
virtual const char* FB_CARG getTriggerTable(Firebird::IStatus* /*status*/) const
|
|
|
|
{
|
|
|
|
return triggerTable.c_str();
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual Firebird::ExternalTrigger::Type FB_CARG getTriggerType(Firebird::IStatus* /*status*/) const
|
|
|
|
{
|
|
|
|
return triggerType;
|
|
|
|
}
|
|
|
|
|
|
|
|
public:
|
|
|
|
Firebird::MetaName package;
|
|
|
|
Firebird::MetaName name;
|
|
|
|
Firebird::string entryPoint;
|
|
|
|
Firebird::string body;
|
2013-02-17 13:08:53 +01:00
|
|
|
Firebird::RefPtr<Firebird::MsgMetadata> inputParameters;
|
|
|
|
Firebird::RefPtr<Firebird::MsgMetadata> outputParameters;
|
|
|
|
Firebird::RefPtr<Firebird::MsgMetadata> triggerFields;
|
2011-05-15 17:37:11 +02:00
|
|
|
Firebird::MetaName triggerTable;
|
|
|
|
Firebird::ExternalTrigger::Type triggerType;
|
2013-02-17 13:08:53 +01:00
|
|
|
|
|
|
|
private:
|
|
|
|
static Firebird::IMessageMetadata* getMetadata(const Firebird::IMessageMetadata* par)
|
|
|
|
{
|
|
|
|
Firebird::IMessageMetadata* rc = const_cast<Firebird::IMessageMetadata*>(par);
|
|
|
|
rc->addRef();
|
|
|
|
return rc;
|
|
|
|
}
|
2011-05-15 17:37:11 +02:00
|
|
|
};
|
|
|
|
|
2009-10-21 02:42:38 +02:00
|
|
|
class ExternalContextImpl : public Firebird::ExternalContext
|
|
|
|
{
|
|
|
|
friend class AttachmentImpl;
|
|
|
|
|
|
|
|
public:
|
|
|
|
ExternalContextImpl(thread_db* tdbb, Firebird::ExternalEngine* aEngine);
|
|
|
|
virtual ~ExternalContextImpl();
|
|
|
|
|
|
|
|
void releaseTransaction();
|
|
|
|
void setTransaction(thread_db* tdbb);
|
|
|
|
|
2013-02-15 03:48:54 +01:00
|
|
|
virtual Firebird::IMaster* FB_CALL getMaster();
|
|
|
|
virtual Firebird::ExternalEngine* FB_CALL getEngine(Firebird::IStatus* status);
|
|
|
|
virtual Firebird::IAttachment* FB_CALL getAttachment(Firebird::IStatus* status);
|
|
|
|
virtual Firebird::ITransaction* FB_CALL getTransaction(Firebird::IStatus* status);
|
2011-05-07 21:52:44 +02:00
|
|
|
virtual const char* FB_CALL getUserName();
|
|
|
|
virtual const char* FB_CALL getDatabaseName();
|
2009-10-21 02:42:38 +02:00
|
|
|
virtual const Firebird::Utf8* FB_CALL getClientCharSet();
|
|
|
|
virtual int FB_CALL obtainInfoCode();
|
|
|
|
virtual void* FB_CALL getInfo(int code);
|
|
|
|
virtual void* FB_CALL setInfo(int code, void* value);
|
|
|
|
|
|
|
|
private:
|
|
|
|
Firebird::ExternalEngine* engine;
|
|
|
|
Attachment* internalAttachment;
|
2011-05-07 21:52:44 +02:00
|
|
|
jrd_tra* internalTransaction;
|
|
|
|
Firebird::IAttachment* externalAttachment;
|
|
|
|
Firebird::ITransaction* externalTransaction;
|
2009-10-21 02:42:38 +02:00
|
|
|
Firebird::GenericMap<Firebird::NonPooled<int, void*> > miscInfo;
|
|
|
|
Firebird::MetaName clientCharSet;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct EngineAttachment
|
|
|
|
{
|
|
|
|
EngineAttachment(Firebird::ExternalEngine* aEngine, Jrd::Attachment* aAttachment)
|
|
|
|
: engine(aEngine),
|
|
|
|
attachment(aAttachment)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool greaterThan(const EngineAttachment& i1, const EngineAttachment& i2)
|
|
|
|
{
|
|
|
|
return (i1.engine > i2.engine) ||
|
|
|
|
(i1.engine == i2.engine && i1.attachment > i2.attachment);
|
|
|
|
}
|
|
|
|
|
|
|
|
Firebird::ExternalEngine* engine;
|
|
|
|
Jrd::Attachment* attachment;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct EngineAttachmentInfo
|
|
|
|
{
|
|
|
|
EngineAttachmentInfo()
|
|
|
|
: engine(NULL),
|
2009-10-30 11:50:59 +01:00
|
|
|
context(NULL),
|
|
|
|
adminCharSet(0)
|
2009-10-21 02:42:38 +02:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
Firebird::ExternalEngine* engine;
|
|
|
|
Firebird::AutoPtr<ExternalContextImpl> context;
|
|
|
|
USHORT adminCharSet;
|
|
|
|
};
|
|
|
|
|
|
|
|
public:
|
|
|
|
class Function
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Function(thread_db* tdbb, ExtEngineManager* aExtManager,
|
|
|
|
Firebird::ExternalEngine* aEngine,
|
2011-05-15 17:37:11 +02:00
|
|
|
RoutineMetadata* aMetadata,
|
2009-10-21 02:42:38 +02:00
|
|
|
Firebird::ExternalFunction* aFunction,
|
2009-12-21 18:43:01 +01:00
|
|
|
const Jrd::Function* aUdf);
|
2009-10-21 02:42:38 +02:00
|
|
|
~Function();
|
|
|
|
|
2011-06-01 03:44:54 +02:00
|
|
|
void execute(thread_db* tdbb, UCHAR* inMsg, UCHAR* outMsg) const;
|
2009-10-21 02:42:38 +02:00
|
|
|
|
|
|
|
private:
|
|
|
|
ExtEngineManager* extManager;
|
|
|
|
Firebird::ExternalEngine* engine;
|
2011-05-15 17:37:11 +02:00
|
|
|
Firebird::AutoPtr<RoutineMetadata> metadata;
|
2009-10-21 02:42:38 +02:00
|
|
|
Firebird::ExternalFunction* function;
|
2009-12-21 18:43:01 +01:00
|
|
|
const Jrd::Function* udf;
|
2009-10-21 02:42:38 +02:00
|
|
|
Database* database;
|
|
|
|
};
|
|
|
|
|
|
|
|
class ResultSet;
|
|
|
|
|
|
|
|
class Procedure
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Procedure(thread_db* tdbb, ExtEngineManager* aExtManager,
|
|
|
|
Firebird::ExternalEngine* aEngine,
|
2011-05-15 17:37:11 +02:00
|
|
|
RoutineMetadata* aMetadata,
|
2009-10-21 02:42:38 +02:00
|
|
|
Firebird::ExternalProcedure* aProcedure,
|
|
|
|
const jrd_prc* aPrc);
|
|
|
|
~Procedure();
|
|
|
|
|
2011-06-01 03:44:54 +02:00
|
|
|
ResultSet* open(thread_db* tdbb, UCHAR* inMsg, UCHAR* outMsg) const;
|
2009-10-21 02:42:38 +02:00
|
|
|
|
|
|
|
private:
|
|
|
|
ExtEngineManager* extManager;
|
|
|
|
Firebird::ExternalEngine* engine;
|
2011-05-15 17:37:11 +02:00
|
|
|
Firebird::AutoPtr<RoutineMetadata> metadata;
|
2009-10-21 02:42:38 +02:00
|
|
|
Firebird::ExternalProcedure* procedure;
|
|
|
|
const jrd_prc* prc;
|
|
|
|
Database* database;
|
|
|
|
|
|
|
|
friend class ResultSet;
|
|
|
|
};
|
|
|
|
|
|
|
|
class ResultSet
|
|
|
|
{
|
|
|
|
public:
|
2011-06-01 03:44:54 +02:00
|
|
|
ResultSet(thread_db* tdbb, UCHAR* inMsg, UCHAR* outMsg, const Procedure* aProcedure);
|
2009-10-21 02:42:38 +02:00
|
|
|
~ResultSet();
|
|
|
|
|
|
|
|
bool fetch(thread_db* tdbb);
|
|
|
|
|
|
|
|
private:
|
2009-12-27 23:05:22 +01:00
|
|
|
const Procedure* procedure;
|
2011-05-09 12:15:19 +02:00
|
|
|
Attachment* attachment;
|
2009-10-21 02:42:38 +02:00
|
|
|
bool firstFetch;
|
|
|
|
EngineAttachmentInfo* attInfo;
|
|
|
|
Firebird::ExternalResultSet* resultSet;
|
|
|
|
USHORT charSet;
|
|
|
|
};
|
|
|
|
|
|
|
|
class Trigger
|
|
|
|
{
|
|
|
|
public:
|
2013-01-29 02:21:32 +01:00
|
|
|
Trigger(thread_db* tdbb, MemoryPool& pool, ExtEngineManager* aExtManager,
|
|
|
|
Firebird::ExternalEngine* aEngine, RoutineMetadata* aMetadata, TriggerMessage& fieldsMsg,
|
|
|
|
Firebird::ExternalTrigger* aTrigger, const Jrd::Trigger* aTrg);
|
2009-10-21 02:42:38 +02:00
|
|
|
~Trigger();
|
|
|
|
|
|
|
|
void execute(thread_db* tdbb, Firebird::ExternalTrigger::Action action,
|
2010-10-12 13:36:51 +02:00
|
|
|
record_param* oldRpb, record_param* newRpb) const;
|
2009-10-21 02:42:38 +02:00
|
|
|
|
|
|
|
private:
|
2011-06-01 03:44:54 +02:00
|
|
|
void setValues(thread_db* tdbb, Firebird::Array<UCHAR>& msgBuffer, record_param* rpb) const;
|
2009-10-21 02:42:38 +02:00
|
|
|
|
|
|
|
ExtEngineManager* extManager;
|
|
|
|
Firebird::ExternalEngine* engine;
|
2011-05-15 17:37:11 +02:00
|
|
|
Firebird::AutoPtr<RoutineMetadata> metadata;
|
2013-01-29 02:21:32 +01:00
|
|
|
Firebird::AutoPtr<Format> format;
|
2009-10-21 02:42:38 +02:00
|
|
|
Firebird::ExternalTrigger* trigger;
|
|
|
|
const Jrd::Trigger* trg;
|
2013-01-29 02:21:32 +01:00
|
|
|
Firebird::Array<USHORT> fieldsPos;
|
2009-10-21 02:42:38 +02:00
|
|
|
Database* database;
|
|
|
|
};
|
|
|
|
|
|
|
|
public:
|
2009-10-30 11:50:59 +01:00
|
|
|
explicit ExtEngineManager(MemoryPool& p)
|
2009-10-21 02:42:38 +02:00
|
|
|
: PermanentStorage(p),
|
|
|
|
engines(p),
|
|
|
|
enginesAttachments(p)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
~ExtEngineManager();
|
|
|
|
|
|
|
|
public:
|
|
|
|
static void initialize();
|
|
|
|
|
|
|
|
public:
|
|
|
|
void closeAttachment(thread_db* tdbb, Attachment* attachment);
|
|
|
|
|
2009-12-21 18:43:01 +01:00
|
|
|
Function* makeFunction(thread_db* tdbb, const Jrd::Function* udf,
|
2009-10-21 02:42:38 +02:00
|
|
|
const Firebird::MetaName& engine, const Firebird::string& entryPoint,
|
2013-01-29 02:21:32 +01:00
|
|
|
const Firebird::string& body, RoutineMessage* inMsg, RoutineMessage* outMsg);
|
2009-10-21 02:42:38 +02:00
|
|
|
Procedure* makeProcedure(thread_db* tdbb, const jrd_prc* prc,
|
|
|
|
const Firebird::MetaName& engine, const Firebird::string& entryPoint,
|
2013-01-29 02:21:32 +01:00
|
|
|
const Firebird::string& body, RoutineMessage* inMsg, RoutineMessage* outMsg);
|
2009-10-21 02:42:38 +02:00
|
|
|
Trigger* makeTrigger(thread_db* tdbb, const Jrd::Trigger* trg,
|
|
|
|
const Firebird::MetaName& engine, const Firebird::string& entryPoint,
|
|
|
|
const Firebird::string& body, Firebird::ExternalTrigger::Type type);
|
|
|
|
|
|
|
|
private:
|
|
|
|
Firebird::ExternalEngine* getEngine(thread_db* tdbb,
|
|
|
|
const Firebird::MetaName& name);
|
|
|
|
EngineAttachmentInfo* getEngineAttachment(thread_db* tdbb,
|
|
|
|
const Firebird::MetaName& name);
|
|
|
|
EngineAttachmentInfo* getEngineAttachment(thread_db* tdbb,
|
|
|
|
Firebird::ExternalEngine* engine, bool closing = false);
|
2009-11-14 23:21:11 +01:00
|
|
|
void setupAdminCharSet(thread_db* tdbb, Firebird::ExternalEngine* engine,
|
|
|
|
EngineAttachmentInfo* attInfo);
|
2009-10-21 02:42:38 +02:00
|
|
|
|
|
|
|
private:
|
|
|
|
typedef Firebird::GenericMap<Firebird::Pair<
|
|
|
|
Firebird::Left<Firebird::MetaName, Firebird::ExternalEngine*> > > EnginesMap;
|
|
|
|
typedef Firebird::GenericMap<Firebird::Pair<Firebird::NonPooled<
|
|
|
|
EngineAttachment, EngineAttachmentInfo*> >, EngineAttachment> EnginesAttachmentsMap;
|
|
|
|
|
|
|
|
Firebird::RWLock enginesLock;
|
|
|
|
EnginesMap engines;
|
|
|
|
EnginesAttachmentsMap enginesAttachments;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
} // namespace Jrd
|
|
|
|
|
|
|
|
#endif // JRD_EXT_ENGINE_MANAGER_H
|