/**
* vim: set ts=4 :
* =============================================================================
* SourceMod PostgreSQL Extension
* Copyright (C) 2013 AlliedModders LLC. All rights reserved.
* =============================================================================
*
* This program is free software; you can redistribute it and/or modify it under
* the terms of the GNU General Public License, version 3.0, as published by the
* Free Software Foundation.
*
* This program is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
* FOR A PARTICULAR PURPOSE. See the GNU General Public License for more
* details.
*
* You should have received a copy of the GNU General Public License along with
* this program. If not, see .
*
* As a special exception, AlliedModders LLC gives you permission to link the
* code of this program (as well as its derivative works) to "Half-Life 2," the
* "Source Engine," the "SourcePawn JIT," and any Game MODs that run on software
* by the Valve Corporation. You must obey the GNU General Public License in
* all respects for all other code used. Additionally, AlliedModders LLC grants
* this exception to all derivative works. AlliedModders LLC defines further
* exceptions, found in LICENSE.txt (as of this writing, version JULY-31-2007),
* or .
*
* Version: $Id$
*/
#include "PgStatement.h"
PgStatement::PgStatement(PgDatabase *db, const char* stmtName)
: m_pgsql(db->m_pgsql), m_pParent(db), m_insertID(0), m_affectedRows(0), m_rs(NULL), m_Results(false)
{
m_stmtName = new char[10];
strncopy(m_stmtName, stmtName, 10);
PGresult *desc = PQdescribePrepared(m_pgsql, m_stmtName);
// TODO: Proper error handling?
if (PQresultStatus(desc) != PGRES_COMMAND_OK)
{
PQclear(desc);
return;
}
m_Params = (unsigned int)PQnparams(desc);
if (m_Params)
{
m_pushinfo = (ParamBind *)malloc(sizeof(ParamBind) * m_Params);
memset(m_pushinfo, 0, sizeof(ParamBind) * m_Params);
} else {
m_pushinfo = NULL;
}
m_Results = false;
}
PgStatement::~PgStatement()
{
/* Free result set structures */
if (m_Results)
{
if (m_rs->m_pRes != NULL)
PQclear(m_rs->m_pRes);
delete m_rs;
}
/* Free old blobs */
for (unsigned int i=0; i= m_Params)
{
return false;
}
m_pushinfo[param].data.ival = num;
m_pushinfo[param].type = DBType_Integer;
return true;
}
bool PgStatement::BindParamFloat(unsigned int param, float f)
{
if (param >= m_Params)
{
return false;
}
m_pushinfo[param].data.fval = f;
m_pushinfo[param].type = DBType_Float;
return true;
}
bool PgStatement::BindParamString(unsigned int param, const char *text, bool copy)
{
if (param >= m_Params)
{
return false;
}
const void *final_ptr;
size_t len;
if (copy)
{
len = strlen(text);
final_ptr = CopyBlob(param, text, len+1);
} else {
len = strlen(text);
final_ptr = text;
}
m_pushinfo[param].blob = (void *)final_ptr;
m_pushinfo[param].length = len;
m_pushinfo[param].type = DBType_String;
return true;
}
bool PgStatement::BindParamBlob(unsigned int param, const void *data, size_t length, bool copy)
{
if (param >= m_Params)
{
return false;
}
const void *final_ptr;
if (copy)
{
final_ptr = CopyBlob(param, data, length);
} else {
final_ptr = data;
}
m_pushinfo[param].blob = (void *)final_ptr;
m_pushinfo[param].length = length;
m_pushinfo[param].type = DBType_Blob;
return true;
}
bool PgStatement::BindParamNull(unsigned int param)
{
if (param >= m_Params)
{
return false;
}
m_pushinfo[param].type = DBType_NULL;
return true;
}
bool PgStatement::Execute()
{
/* Clear any past result first! */
if (m_Results)
delete m_rs;
m_Results = false;
PGresult *res;
/* Bind the parameters */
if (m_Params)
{
// Put bound params into a nice array
const char **paramValues = new const char*[m_Params];
int *paramLengths = new int[m_Params];
int *paramFormats = new int[m_Params];
for (unsigned int i=0; iSetLastIDAndRows(m_insertID, m_affectedRows);
/* Skip away if we don't have data */
if (status == PGRES_COMMAND_OK)
{
PQclear(res);
return true;
}
m_rs = new PgBasicResults(res);
m_Results = true;
return true;
}
const char *PgStatement::GetError(int *errCode/* =NULL */)
{
if (m_Results)
{
if (errCode)
{
// PostgreSQL only supports SQLSTATE error codes.
// https://www.postgresql.org/docs/9.6/errcodes-appendix.html
*errCode = -1;
}
return PQresultErrorMessage(m_rs->m_pRes);
}
else
{
return PQerrorMessage(m_pgsql);
}
}
unsigned int PgStatement::GetAffectedRows()
{
return m_affectedRows;
}
unsigned int PgStatement::GetInsertID()
{
return m_insertID;
}
IResultSet *PgStatement::GetResultSet()
{
return (m_Results ? m_rs : NULL);
}