/* * 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 * and all contributors signed below. * * All Rights Reserved. * Contributor(s): ______________________________________. * * Paul Reeves, 2023 * */ #define FB_UDR_STATUS_TYPE ::Firebird::ThrowStatusWrapper //#include //#include #include "MyFirstUDRKit.h" #include #include #include #include #include #include #ifdef HAVE_MATH_H #include #endif #ifdef HAVE_STRING_H #include #endif #ifdef TIME_WITH_SYS_TIME # include # include #else # ifdef HAVE_SYS_TIME_H # include # else # include # endif #endif #ifdef HAVE_SYS_TIMEB_H # include #endif #ifdef HAVE_LOCALE_H #include #endif #include using namespace std; using namespace Firebird; // First, let's start with a function from the UdfBackwardCompatibility // library and add some additional comments. // We will also use self-docmenting variable names // ////////////////////////////////////////////////// /*** * create function div ( * anumerator integer, * adenominator integer * ) returns double precision * external name 'MyFirstUDRKit!MFK_div!Divide anumerator by adenominator' * engine udr; ***/ FB_UDR_BEGIN_FUNCTION(MFK_div) // Divide integer anumerator 1 by integer adenominator using the stdlib FB_UDR_MESSAGE(InMessage, (FB_INTEGER, anumerator) (FB_INTEGER, adenominator) ); FB_UDR_MESSAGE(OutMessage, (FB_DOUBLE, result) ); FB_UDR_EXECUTE_FUNCTION { if (in->anumeratorNull || in->adenominatorNull) { out->resultNull = FB_TRUE; out->result = 0; } else { out->resultNull = FB_FALSE; if (in->adenominator) { out->result = div(in->anumerator, in->adenominator).quot; } else { // This is old style - surely we can do better? out->result = std::numeric_limits::infinity(); ISC_STATUS_ARRAY StatusVector = {isc_arg_gds, isc_arith_except, isc_arg_gds, isc_exception_integer_divide_by_zero, isc_arg_end}; FbException::check(isc_exception_integer_divide_by_zero, status, StatusVector); } } } FB_UDR_END_FUNCTION /*** DDL create function flagged ( flags integer, flag integer ) returns integer external name 'MyFirstUDRKit!MFK_flagged!How is this function used?' engine udr; ***/ FB_UDR_BEGIN_FUNCTION(MFK_flagged) FB_UDR_MESSAGE(InMessage, (FB_INTEGER, flags) (FB_INTEGER, flag) ); FB_UDR_MESSAGE(OutMessage, (FB_INTEGER, result) ); FB_UDR_EXECUTE_FUNCTION { // Original code // if ( flags == NULL ) { // return 0; // } if ( in->flagsNull != 0 ) { out->resultNull = FB_TRUE; out->result = 0; } else { out->resultNull = FB_FALSE; // Original code // ISC_UINT64 i = ( 1ULL << *flag ); // return ( *flags & i ) ? 1 : 0; ISC_UINT64 i = ( 1ULL << in->flag ); out->result = ( in->flags & i ) ? 1 : 0; } } FB_UDR_END_FUNCTION /*** DDL * create or alter function LoadBlobFromFile ( * afilename varchar(8191), * ablob BLOB * ) returns bigint * external name 'MyFirstUDRKit!MFKLoadBlobFromFile!Load file and save to Blob' * engine udr; ***/ FB_UDR_BEGIN_FUNCTION (MFK_LoadBlobFromFile) //BEGIN FB_UDR_MESSAGE(InMessage, (FB_CHAR(8191), afilename) (FB_BLOB, ablob) ); FB_UDR_MESSAGE(OutMessage, (FB_BIGINT, result) ); AutoRelease att; AutoRelease tra; AutoRelease Blob; FB_UDR_EXECUTE_FUNCTION { // Test Input if (in->afilenameNull != 0) { out->resultNull = FB_TRUE; out->result = 0; } ifstream File ( in->afilename.str , ios_base::in | ios_base::binary ); if (! File.is_open()) { out->result = -2; return; } // This is wrong - the calling application should pass in a blob handle AutoRelease Blob(att->createBlob(status, tra, &in->ablob, 0, nullptr)); if (Blob == nullptr) { out->result = -1; return; } vector Buffer (MaxSegmentSize, 0); streamsize DataSize; while ( File.good() ) { File.read( Buffer.data(), Buffer.size() ); DataSize = File.gcount(); Blob->putSegment( status, DataSize, Buffer.data() ); out->result += DataSize; // Perhaps test for badbit here? } if (File.bad()) { // Something went wrong // What to do? out->resultNull = FB_TRUE; // What do we do with the partially written blob? Is cancel enough? Blob->cancel( status ); // Should we reset result to 0? } Blob->close( status ); Blob->release(); Buffer.clear(); File.close(); } FB_UDR_END_FUNCTION //END /*** create or alter function SaveBlobToFile ( afilename varchar(8191), ablob BLOB ) returns bigint external name 'MyFirstUDRKit!MFK_SaveBlobToFile!Save blob to file' engine udr; ***/ FB_UDR_BEGIN_FUNCTION(MFK_SaveBlobToFile) //BEGIN FB_UDR_MESSAGE(InMessage, (FB_CHAR(8191), afilename) (FB_BLOB, ablob) ); FB_UDR_MESSAGE(OutMessage, (FB_BIGINT, result) ); AutoRelease att; AutoRelease tra; AutoRelease Blob; FB_UDR_EXECUTE_FUNCTION { // Test Input if (in->afilenameNull != 0 || in->ablobNull != 0 ) { out->resultNull = FB_TRUE; out->result = 0; } /* * DOC NOTE - We do not test for existence of the file here! */ fstream File ( in->afilename.str , fstream::out | fstream::binary | fstream::app ); if (! File.is_open()) { out->result = -2; return; } Blob.reset(att->openBlob(status, tra, &in->ablob, 0, nullptr)); if (Blob == nullptr) { out->result = -1; return; } vector Buffer (MaxSegmentSize, 0); unsigned BytesRead = 0; for (bool Eof = false; !Eof; ) { switch (Blob->getSegment( status, MaxSegmentSize, Buffer.data(), &BytesRead)) { case IStatus::RESULT_OK: case IStatus::RESULT_SEGMENT: { File.write( Buffer.data(), Buffer.size() ); out->result += BytesRead; continue; } default: { Blob->close( status ); // will close interface Blob.release(); Eof = true; break; } } } Buffer.clear(); File.close(); } FB_UDR_END_FUNCTION //END /*** DDL create function BillDate ( d integer, date ) returns date external name 'my_first_udr_kit!MFK_BillDate!What does BillDate do?' engine udr; ***/ /* FB_UDR_BEGIN_FUNCTION(MFK_BillDate) FB_UDR_MESSAGE(InMessage, (FB_INTEGER, d) (FB_INTEGER, date) ); FB_UDR_MESSAGE(OutMessage, (FB_DATE, result) ); FB_UDR_END_FUNCTION */