2013-06-25 15:37:03 +02:00
|
|
|
/*
|
|
|
|
* PROGRAM: Object oriented API samples.
|
|
|
|
* MODULE: 04.print_table.cpp
|
|
|
|
* DESCRIPTION: Run SELECT statement without parameters.
|
|
|
|
* Use attachment method to open cursor.
|
|
|
|
* Print all fields for selected records in a table.
|
|
|
|
* Learns how to access blob data and use unprepared statement.
|
|
|
|
*
|
|
|
|
* Example for the following interfaces:
|
|
|
|
*
|
|
|
|
* IAttachment - database attachment
|
|
|
|
* IMessageMetadata - describe input and output data format
|
|
|
|
* IResultSet - fetch data returned by statement after execution
|
|
|
|
*
|
|
|
|
* 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 Alexander Peshkoff
|
|
|
|
* for the Firebird Open Source RDBMS project.
|
|
|
|
*
|
|
|
|
* Copyright (c) 2013 Alexander Peshkoff <peshkoff@mail.ru>
|
|
|
|
* and all contributors signed below.
|
|
|
|
*
|
|
|
|
* All Rights Reserved.
|
|
|
|
* Contributor(s): ______________________________________.
|
|
|
|
*/
|
|
|
|
|
2014-12-18 17:54:44 +01:00
|
|
|
#include "ifaceExamples.h"
|
2013-06-25 15:37:03 +02:00
|
|
|
|
|
|
|
static IMaster* master = fb_get_master_interface();
|
|
|
|
|
|
|
|
struct MyField
|
|
|
|
{
|
|
|
|
const char* name;
|
|
|
|
unsigned type, length, offset, null;
|
|
|
|
|
2015-02-10 14:32:03 +01:00
|
|
|
void print(ThrowStatusWrapper* st, IAttachment* att, ITransaction* tra, unsigned char* buf);
|
2013-06-25 15:37:03 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
int main()
|
|
|
|
{
|
|
|
|
int rc = 0;
|
2013-06-28 04:03:10 +02:00
|
|
|
char s[100];
|
2013-06-25 15:37:03 +02:00
|
|
|
|
|
|
|
setenv("ISC_USER", "sysdba", 0);
|
|
|
|
setenv("ISC_PASSWORD", "masterkey", 0);
|
|
|
|
|
2015-11-05 18:15:10 +01:00
|
|
|
ThrowStatusWrapper status(master->getStatus());
|
2013-06-25 15:37:03 +02:00
|
|
|
IProvider* prov = master->getDispatcher();
|
|
|
|
|
|
|
|
IAttachment* att = NULL;
|
|
|
|
ITransaction* tra = NULL;
|
|
|
|
IResultSet* curs = NULL;
|
|
|
|
IMessageMetadata* meta = NULL;
|
|
|
|
|
|
|
|
try
|
|
|
|
{
|
2015-02-10 14:32:03 +01:00
|
|
|
att = prov->attachDatabase(&status, "employee", 0, NULL);
|
|
|
|
tra = att->startTransaction(&status, 0, NULL);
|
2013-06-25 15:37:03 +02:00
|
|
|
|
|
|
|
// If we are not going to run same SELECT query many times we may do not prepare it,
|
|
|
|
// opening cursor instead with single API call.
|
|
|
|
// If statement has input parameters and we know them, appropriate IMetadata may be
|
|
|
|
// constructed and passed to openCursor() together with data buffer.
|
|
|
|
// We also may provide out format info and coerce data passing appropriate metadata
|
|
|
|
// into API call.
|
|
|
|
// In this sample we have no input parameters and do not coerce anything - just
|
|
|
|
// print what we get from SQL query.
|
|
|
|
const char* sql = "select * from rdb$relations where RDB$RELATION_ID < 3 "
|
2013-06-26 09:17:20 +02:00
|
|
|
"or RDB$VIEW_SOURCE is not null";
|
2013-06-25 15:37:03 +02:00
|
|
|
|
|
|
|
// Do not use IStatement - just ask attachment to open cursor
|
2016-03-18 16:56:47 +01:00
|
|
|
curs = att->openCursor(&status, tra, 0, sql, SAMPLES_DIALECT, NULL, NULL, NULL, NULL, 0);
|
2015-02-10 14:32:03 +01:00
|
|
|
meta = curs->getMetadata(&status);
|
|
|
|
unsigned cols = meta->getCount(&status);
|
2013-06-25 15:37:03 +02:00
|
|
|
|
|
|
|
MyField* fields = new MyField[cols];
|
|
|
|
memset(fields, 0, sizeof(MyField) * cols);
|
|
|
|
|
|
|
|
unsigned f = 0;
|
|
|
|
for (unsigned j = 0; j < cols; ++j)
|
|
|
|
{
|
2015-02-10 14:32:03 +01:00
|
|
|
unsigned t = meta->getType(&status, j) & ~1;
|
|
|
|
unsigned sub = meta->getSubType(&status, j);
|
2013-06-25 15:37:03 +02:00
|
|
|
|
2013-06-28 04:03:10 +02:00
|
|
|
switch (t)
|
2013-06-25 15:37:03 +02:00
|
|
|
{
|
|
|
|
case SQL_BLOB:
|
|
|
|
if (sub != 1)
|
|
|
|
continue;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SQL_TEXT:
|
|
|
|
case SQL_VARYING:
|
|
|
|
case SQL_SHORT:
|
|
|
|
case SQL_DOUBLE:
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
{
|
2015-02-10 14:32:03 +01:00
|
|
|
sprintf(s, "Unknown type %d for %s", t, meta->getField(&status, j));
|
2013-06-25 15:37:03 +02:00
|
|
|
throw s;
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// we can work with this field - cache metadata info for fast access
|
|
|
|
fields[f].type = t;
|
2015-02-10 14:32:03 +01:00
|
|
|
fields[f].name = meta->getField(&status, j);
|
|
|
|
fields[f].length = meta->getLength(&status, j);
|
|
|
|
fields[f].offset = meta->getOffset(&status, j);
|
|
|
|
fields[f].null = meta->getNullOffset(&status, j);
|
2013-06-25 15:37:03 +02:00
|
|
|
++f;
|
|
|
|
}
|
|
|
|
|
2015-02-10 14:32:03 +01:00
|
|
|
unsigned l = meta->getMessageLength(&status);
|
2013-06-25 15:37:03 +02:00
|
|
|
unsigned char* buffer = new unsigned char[l];
|
|
|
|
|
|
|
|
// fetch records from cursor
|
2015-02-27 16:42:53 +01:00
|
|
|
while (curs->fetchNext(&status, buffer) == IStatus::RESULT_OK)
|
2013-06-25 15:37:03 +02:00
|
|
|
{
|
|
|
|
for (unsigned j = 0; j < f; ++j)
|
|
|
|
{
|
|
|
|
// call field's function to print it
|
2015-02-10 14:32:03 +01:00
|
|
|
fields[j].print(&status, att, tra, buffer);
|
2013-06-25 15:37:03 +02:00
|
|
|
}
|
|
|
|
printf("\n");
|
|
|
|
}
|
|
|
|
|
2015-02-10 14:32:03 +01:00
|
|
|
curs->close(&status);
|
2013-06-25 15:37:03 +02:00
|
|
|
curs = NULL;
|
|
|
|
|
|
|
|
meta->release();
|
|
|
|
meta = NULL;
|
|
|
|
|
2015-02-10 14:32:03 +01:00
|
|
|
tra->commit(&status);
|
2013-06-25 15:37:03 +02:00
|
|
|
tra = NULL;
|
|
|
|
|
2015-02-10 14:32:03 +01:00
|
|
|
att->detach(&status);
|
2013-06-25 15:37:03 +02:00
|
|
|
att = NULL;
|
|
|
|
}
|
2015-02-27 16:59:11 +01:00
|
|
|
catch (const FbException& error)
|
2013-06-25 15:37:03 +02:00
|
|
|
{
|
2015-02-27 16:59:11 +01:00
|
|
|
// handle error
|
2013-06-25 15:37:03 +02:00
|
|
|
fflush(stdout);
|
2015-02-27 16:59:11 +01:00
|
|
|
rc = 1;
|
2015-11-05 18:15:10 +01:00
|
|
|
|
|
|
|
char buf[256];
|
|
|
|
master->getUtilInterface()->formatStatus(buf, sizeof(buf), error.getStatus());
|
|
|
|
fprintf(stderr, "%s\n", buf);
|
2013-06-25 15:37:03 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (meta)
|
|
|
|
meta->release();
|
|
|
|
if (curs)
|
|
|
|
curs->release();
|
|
|
|
if (tra)
|
|
|
|
tra->release();
|
|
|
|
if (att)
|
|
|
|
att->release();
|
2015-11-05 18:15:10 +01:00
|
|
|
|
|
|
|
prov->release();
|
|
|
|
status.dispose();
|
2013-06-25 15:37:03 +02:00
|
|
|
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
T as(unsigned char* ptr)
|
|
|
|
{
|
|
|
|
return *((T*) ptr);
|
|
|
|
}
|
|
|
|
|
2015-02-10 14:32:03 +01:00
|
|
|
void MyField::print(ThrowStatusWrapper* st, IAttachment* att, ITransaction* tra, unsigned char* buf)
|
2013-06-25 15:37:03 +02:00
|
|
|
{
|
|
|
|
printf("%s: ", name);
|
|
|
|
if (as<short>(buf + null))
|
|
|
|
{
|
|
|
|
printf("<Null>\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// IBlob makes it possible to read/write BLOB data
|
|
|
|
IBlob* blob = NULL;
|
2013-06-28 04:03:10 +02:00
|
|
|
switch (type)
|
2013-06-25 15:37:03 +02:00
|
|
|
{
|
|
|
|
// text fields
|
|
|
|
case SQL_TEXT:
|
|
|
|
printf("%*.*s\n", length, length, buf + offset);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SQL_VARYING:
|
|
|
|
{
|
|
|
|
unsigned l = as<short>(buf + offset);
|
|
|
|
printf("%*.*s\n", l, l, buf + offset + sizeof(short));
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
// numeric fields
|
|
|
|
case SQL_SHORT:
|
|
|
|
printf("%d\n", as<short>(buf + offset));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SQL_DOUBLE:
|
|
|
|
printf("%f\n", as<double>(buf + offset));
|
|
|
|
break;
|
|
|
|
|
|
|
|
// blob requires more handling in DB
|
|
|
|
case SQL_BLOB:
|
|
|
|
try
|
|
|
|
{
|
|
|
|
// use attachment's method to access BLOB object
|
2014-09-29 13:03:47 +02:00
|
|
|
blob = att->openBlob(st, tra, (ISC_QUAD*) (buf + offset), 0, NULL);
|
2013-06-25 15:37:03 +02:00
|
|
|
|
|
|
|
char segbuf[16];
|
|
|
|
unsigned len;
|
|
|
|
// read data segment by segment
|
2014-10-12 04:25:02 +02:00
|
|
|
for (;;)
|
2013-06-25 15:37:03 +02:00
|
|
|
{
|
2014-08-27 11:24:30 +02:00
|
|
|
int cc = blob->getSegment(st, sizeof(segbuf), segbuf, &len);
|
2015-02-27 16:42:53 +01:00
|
|
|
if (cc != IStatus::RESULT_OK && cc != IStatus::RESULT_SEGMENT)
|
2014-08-27 11:24:30 +02:00
|
|
|
break;
|
2013-06-25 15:37:03 +02:00
|
|
|
fwrite(segbuf, sizeof(char), len, stdout);
|
|
|
|
}
|
|
|
|
|
|
|
|
// close BLOB after receiving all data
|
|
|
|
blob->close(st);
|
|
|
|
blob = NULL;
|
|
|
|
printf("\n");
|
|
|
|
}
|
2013-06-28 04:03:10 +02:00
|
|
|
catch (...)
|
2013-06-25 15:37:03 +02:00
|
|
|
{
|
|
|
|
if (blob)
|
|
|
|
blob->release();
|
|
|
|
throw;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
// do not plan to have all types here
|
|
|
|
default:
|
|
|
|
throw "Unknown type in MyField::print()";
|
|
|
|
}
|
|
|
|
}
|