Skip to content
Fetching contributors…
Cannot retrieve contributors at this time
645 lines (594 sloc) 20 KB
#include <string.h>
#include <stdio.h>
#include <math.h>
#include <libpq-fe.h>
#include <ruby.h>
#if defined(HAVE_RUBY_ENCODING_H) && HAVE_RUBY_ENCODING_H
#define SPG_ENCODING 1
#include <ruby/encoding.h>
#define ENC_INDEX ,enc_index
#else
#define ENC_INDEX
#endif
#define SPG_MAX_FIELDS 256
#define SPG_MILLISECONDS_PER_DAY 86400000000.0
#define SPG_MINUTES_PER_DAY 1440.0
#define SPG_SECONDS_PER_DAY 86400.0
#define SPG_DT_ADD_USEC if (usec != 0) { dt = rb_funcall(dt, spg_id_op_plus, 1, rb_float_new(usec/SPG_MILLISECONDS_PER_DAY)); }
#define SPG_NO_TZ 0
#define SPG_DB_LOCAL 1
#define SPG_DB_UTC 2
#define SPG_APP_LOCAL 4
#define SPG_APP_UTC 8
#define SPG_YIELD_NORMAL 0
#define SPG_YIELD_COLUMN 1
#define SPG_YIELD_COLUMNS 2
#define SPG_YIELD_FIRST 3
#define SPG_YIELD_ARRAY 4
#define SPG_YIELD_KV_HASH 5
#define SPG_YIELD_MKV_HASH 6
#define SPG_YIELD_KMV_HASH 7
#define SPG_YIELD_MKMV_HASH 8
#define SPG_YIELD_MODEL 9
static VALUE spg_Sequel;
static VALUE spg_Blob;
static VALUE spg_BigDecimal;
static VALUE spg_Date;
static VALUE spg_SQLTime;
static VALUE spg_sym_utc;
static VALUE spg_sym_local;
static VALUE spg_sym_map;
static VALUE spg_sym_first;
static VALUE spg_sym_array;
static VALUE spg_sym_hash;
static VALUE spg_sym_model;
static VALUE spg_sym__sequel_pg_type;
static VALUE spg_sym__sequel_pg_value;
static ID spg_id_new;
static ID spg_id_local;
static ID spg_id_year;
static ID spg_id_month;
static ID spg_id_day;
static ID spg_id_output_identifier;
static ID spg_id_datetime_class;
static ID spg_id_application_timezone;
static ID spg_id_timezone;
static ID spg_id_op_plus;
static ID spg_id_utc;
static ID spg_id_utc_offset;
static ID spg_id_localtime;
static ID spg_id_new_offset;
static ID spg_id_call;
static ID spg_id_get;
static ID spg_id_opts;
static ID spg_id_db;
static ID spg_id_conversion_procs;
static ID spg_id_columns;
static ID spg_id_encoding;
static ID spg_id_values;
#if SPG_ENCODING
static int enc_get_index(VALUE val)
{
int i = ENCODING_GET_INLINED(val);
if (i == ENCODING_INLINE_MAX) {
i = NUM2INT(rb_ivar_get(val, spg_id_encoding));
}
return i;
}
#endif
static VALUE spg_time(const char *s) {
VALUE now;
int hour, minute, second, tokens;
char subsec[7];
int usec = 0;
tokens = sscanf(s, "%2d:%2d:%2d.%s", &hour, &minute, &second, subsec);
if(tokens == 4) {
usec = atoi(subsec);
usec *= (int) pow(10, (6 - strlen(subsec)));
} else if(tokens < 3) {
rb_raise(rb_eArgError, "unexpected time format");
}
now = rb_funcall(spg_SQLTime, spg_id_new, 0);
return rb_funcall(spg_SQLTime, spg_id_local, 7, rb_funcall(now, spg_id_year, 0), rb_funcall(now, spg_id_month, 0), rb_funcall(now, spg_id_day, 0), INT2NUM(hour), INT2NUM(minute), INT2NUM(second), INT2NUM(usec));
}
static VALUE spg_date(const char *s) {
int year, month, day;
if(3 != sscanf(s, "%d-%2d-%2d", &year, &month, &day)) {
rb_raise(rb_eArgError, "unexpected date format");
}
return rb_funcall(spg_Date, spg_id_new, 3, INT2NUM(year), INT2NUM(month), INT2NUM(day));
}
static VALUE spg_timestamp(const char *s, VALUE self) {
VALUE dtc, dt, rtz;
int tz = SPG_NO_TZ;
int year, month, day, hour, min, sec, usec, tokens, utc_offset;
int usec_start, usec_stop;
char offset_sign = 0;
int offset_hour = 0;
int offset_minute = 0;
int offset_seconds = 0;
double offset_fraction = 0.0;
if (0 != strchr(s, '.')) {
tokens = sscanf(s, "%d-%2d-%2d %2d:%2d:%2d.%n%d%n%c%02d:%02d",
&year, &month, &day, &hour, &min, &sec,
&usec_start, &usec, &usec_stop,
&offset_sign, &offset_hour, &offset_minute);
if(tokens < 7) {
rb_raise(rb_eArgError, "unexpected datetime format");
}
usec *= (int) pow(10, (6 - (usec_stop - usec_start)));
} else {
tokens = sscanf(s, "%d-%2d-%2d %2d:%2d:%2d%c%02d:%02d",
&year, &month, &day, &hour, &min, &sec,
&offset_sign, &offset_hour, &offset_minute);
if (tokens == 3) {
hour = 0;
min = 0;
sec = 0;
} else if (tokens < 6) {
rb_raise(rb_eArgError, "unexpected datetime format");
}
usec = 0;
}
if (offset_sign == '-') {
offset_hour *= -1;
offset_minute *= -1;
}
/* Get values of datetime_class, database_timezone, and application_timezone */
dtc = rb_funcall(spg_Sequel, spg_id_datetime_class, 0);
rtz = rb_funcall(rb_funcall(self, spg_id_db, 0), spg_id_timezone, 0);
if (rtz == spg_sym_local) {
tz += SPG_DB_LOCAL;
} else if (rtz == spg_sym_utc) {
tz += SPG_DB_UTC;
}
rtz = rb_funcall(spg_Sequel, spg_id_application_timezone, 0);
if (rtz == spg_sym_local) {
tz += SPG_APP_LOCAL;
} else if (rtz == spg_sym_utc) {
tz += SPG_APP_UTC;
}
if (dtc == rb_cTime) {
if (offset_sign) {
/* Offset given, convert to local time if not already in local time.
* While PostgreSQL generally returns timestamps in local time, it's unwise to rely on this.
*/
dt = rb_funcall(rb_cTime, spg_id_local, 7, INT2NUM(year), INT2NUM(month), INT2NUM(day), INT2NUM(hour), INT2NUM(min), INT2NUM(sec), INT2NUM(usec));
utc_offset = NUM2INT(rb_funcall(dt, spg_id_utc_offset, 0));
offset_seconds = offset_hour * 3600 + offset_minute * 60;
if (utc_offset != offset_seconds) {
dt = rb_funcall(dt, spg_id_op_plus, 1, INT2NUM(utc_offset - offset_seconds));
}
if (tz & SPG_APP_UTC) {
dt = rb_funcall(dt, spg_id_utc, 0);
}
return dt;
} else if (tz == SPG_NO_TZ) {
return rb_funcall(rb_cTime, spg_id_local, 7, INT2NUM(year), INT2NUM(month), INT2NUM(day), INT2NUM(hour), INT2NUM(min), INT2NUM(sec), INT2NUM(usec));
}
/* No offset given, and some timezone combination given */
if (tz & SPG_DB_UTC) {
dt = rb_funcall(rb_cTime, spg_id_utc, 7, INT2NUM(year), INT2NUM(month), INT2NUM(day), INT2NUM(hour), INT2NUM(min), INT2NUM(sec), INT2NUM(usec));
if (tz & SPG_APP_LOCAL) {
return rb_funcall(dt, spg_id_localtime, 0);
} else {
return dt;
}
} else {
dt = rb_funcall(rb_cTime, spg_id_local, 7, INT2NUM(year), INT2NUM(month), INT2NUM(day), INT2NUM(hour), INT2NUM(min), INT2NUM(sec), INT2NUM(usec));
if (tz & SPG_APP_UTC) {
return rb_funcall(dt, spg_id_utc, 0);
} else {
return dt;
}
}
} else {
/* datetime.class == DateTime */
if (offset_sign) {
/* Offset given, handle correct local time.
* While PostgreSQL generally returns timestamps in local time, it's unwise to rely on this.
*/
offset_fraction = offset_hour/24.0 + offset_minute/SPG_MINUTES_PER_DAY;
dt = rb_funcall(dtc, spg_id_new, 7, INT2NUM(year), INT2NUM(month), INT2NUM(day), INT2NUM(hour), INT2NUM(min), INT2NUM(sec), rb_float_new(offset_fraction));
SPG_DT_ADD_USEC
if (tz & SPG_APP_LOCAL) {
utc_offset = NUM2INT(rb_funcall(rb_funcall(rb_cTime, spg_id_new, 0), spg_id_utc_offset, 0))/SPG_SECONDS_PER_DAY;
dt = rb_funcall(dt, spg_id_new_offset, 1, rb_float_new(utc_offset));
} else if (tz & SPG_APP_UTC) {
dt = rb_funcall(dt, spg_id_new_offset, 1, INT2NUM(0));
}
return dt;
} else if (tz == SPG_NO_TZ) {
dt = rb_funcall(dtc, spg_id_new, 6, INT2NUM(year), INT2NUM(month), INT2NUM(day), INT2NUM(hour), INT2NUM(min), INT2NUM(sec));
SPG_DT_ADD_USEC
return dt;
}
/* No offset given, and some timezone combination given */
if (tz & SPG_DB_LOCAL) {
offset_fraction = NUM2INT(rb_funcall(rb_funcall(rb_cTime, spg_id_local, 6, INT2NUM(year), INT2NUM(month), INT2NUM(day), INT2NUM(hour), INT2NUM(min), INT2NUM(sec)), spg_id_utc_offset, 0))/SPG_SECONDS_PER_DAY;
dt = rb_funcall(dtc, spg_id_new, 7, INT2NUM(year), INT2NUM(month), INT2NUM(day), INT2NUM(hour), INT2NUM(min), INT2NUM(sec), rb_float_new(offset_fraction));
SPG_DT_ADD_USEC
if (tz & SPG_APP_UTC) {
return rb_funcall(dt, spg_id_new_offset, 1, INT2NUM(0));
} else {
return dt;
}
} else {
dt = rb_funcall(dtc, spg_id_new, 6, INT2NUM(year), INT2NUM(month), INT2NUM(day), INT2NUM(hour), INT2NUM(min), INT2NUM(sec));
SPG_DT_ADD_USEC
if (tz & SPG_APP_LOCAL) {
offset_fraction = NUM2INT(rb_funcall(rb_funcall(rb_cTime, spg_id_local, 6, INT2NUM(year), INT2NUM(month), INT2NUM(day), INT2NUM(hour), INT2NUM(min), INT2NUM(sec)), spg_id_utc_offset, 0))/SPG_SECONDS_PER_DAY;
return rb_funcall(dt, spg_id_new_offset, 1, rb_float_new(offset_fraction));
} else {
return dt;
}
}
}
}
static VALUE spg_fetch_rows_set_cols(VALUE self, VALUE ignore) {
return self;
}
static VALUE spg__col_value(VALUE self, PGresult *res, long i, long j, VALUE* colconvert
#ifdef SPG_ENCODING
, int enc_index
#endif
) {
char *v;
VALUE rv;
size_t l;
if(PQgetisnull(res, i, j)) {
rv = Qnil;
} else {
v = PQgetvalue(res, i, j);
switch(PQftype(res, j)) {
case 16: /* boolean */
rv = *v == 't' ? Qtrue : Qfalse;
break;
case 17: /* bytea */
v = (char *)PQunescapeBytea((unsigned char*)v, &l);
rv = rb_funcall(spg_Blob, spg_id_new, 1, rb_str_new(v, l));
PQfreemem(v);
break;
case 20: /* integer */
case 21:
case 22:
case 23:
case 26:
rv = rb_cstr2inum(v, 10);
break;
case 700: /* float */
case 701:
rv = rb_float_new(rb_cstr_to_dbl(v, Qfalse));
break;
case 790: /* numeric */
case 1700:
rv = rb_funcall(spg_BigDecimal, spg_id_new, 1, rb_str_new(v, PQgetlength(res, i, j)));
break;
case 1082: /* date */
rv = spg_date(v);
break;
case 1083: /* time */
case 1266:
rv = spg_time(v);
break;
case 1114: /* timestamp */
case 1184:
rv = spg_timestamp(v, self);
break;
case 18: /* char */
case 25: /* text */
case 1043: /* varchar*/
rv = rb_tainted_str_new(v, PQgetlength(res, i, j));
#ifdef SPG_ENCODING
rb_enc_associate_index(rv, enc_index);
#endif
break;
default:
rv = rb_tainted_str_new(v, PQgetlength(res, i, j));
#ifdef SPG_ENCODING
rb_enc_associate_index(rv, enc_index);
#endif
if (colconvert[j] != Qnil) {
rv = rb_funcall(colconvert[j], spg_id_call, 1, rv);
}
}
}
return rv;
}
static VALUE spg__col_values(VALUE self, VALUE v, VALUE *colsyms, long nfields, PGresult *res, long i, VALUE *colconvert
#ifdef SPG_ENCODING
, int enc_index
#endif
) {
long j;
VALUE cur;
long len = RARRAY_LEN(v);
VALUE a = rb_ary_new2(len);
for (j=0; j<len; j++) {
cur = rb_ary_entry(v, j);
rb_ary_store(a, j, cur == Qnil ? Qnil : spg__col_value(self, res, i, NUM2LONG(cur), colconvert ENC_INDEX));
}
return a;
}
static long spg__field_id(VALUE v, VALUE *colsyms, long nfields) {
long j;
for (j=0; j<nfields; j++) {
if (colsyms[j] == v) {
return j;
}
}
return -1;
}
static VALUE spg__field_ids(VALUE v, VALUE *colsyms, long nfields) {
long i;
long j;
VALUE cur;
long len = RARRAY_LEN(v);
VALUE pg_columns = rb_ary_new2(len);
for (i=0; i<len; i++) {
cur = rb_ary_entry(v, i);
j = spg__field_id(cur, colsyms, nfields);
rb_ary_store(pg_columns, i, j == -1 ? Qnil : LONG2NUM(j));
}
return pg_columns;
}
static VALUE spg_yield_hash_rows(VALUE self, VALUE rres, VALUE ignore) {
PGresult *res;
VALUE colsyms[SPG_MAX_FIELDS];
VALUE colconvert[SPG_MAX_FIELDS];
long ntuples;
long nfields;
long i;
long j;
VALUE h;
VALUE conv_procs = 0;
VALUE opts;
VALUE pg_type;
VALUE pg_value;
char type = SPG_YIELD_NORMAL;
#ifdef SPG_ENCODING
int enc_index;
enc_index = enc_get_index(rres);
#endif
Data_Get_Struct(rres, PGresult, res);
ntuples = PQntuples(res);
nfields = PQnfields(res);
if (nfields > SPG_MAX_FIELDS) {
rb_raise(rb_eRangeError, "more than %d columns in query", SPG_MAX_FIELDS);
}
for(j=0; j<nfields; j++) {
colsyms[j] = rb_funcall(self, spg_id_output_identifier, 1, rb_str_new2(PQfname(res, j)));
i = PQftype(res, j);
switch (i) {
case 16:
case 17:
case 20:
case 21:
case 22:
case 23:
case 26:
case 700:
case 701:
case 790:
case 1700:
case 1082:
case 1083:
case 1266:
case 1114:
case 1184:
case 18:
case 25:
case 1043:
colconvert[j] = Qnil;
break;
default:
if (conv_procs == 0) {
conv_procs = rb_funcall(rb_funcall(self, spg_id_db, 0), spg_id_conversion_procs, 0);
}
colconvert[j] = rb_funcall(conv_procs, spg_id_get, 1, INT2NUM(i));
break;
}
}
rb_ivar_set(self, spg_id_columns, rb_ary_new4(nfields, colsyms));
opts = rb_funcall(self, spg_id_opts, 0);
if (rb_type(opts) == T_HASH) {
pg_type = rb_hash_aref(opts, spg_sym__sequel_pg_type);
pg_value = rb_hash_aref(opts, spg_sym__sequel_pg_value);
if (SYMBOL_P(pg_type)) {
if (pg_type == spg_sym_map) {
if (SYMBOL_P(pg_value)) {
type = SPG_YIELD_COLUMN;
} else if (rb_type(pg_value) == T_ARRAY) {
type = SPG_YIELD_COLUMNS;
}
} else if (pg_type == spg_sym_first) {
type = SPG_YIELD_FIRST;
} else if (pg_type == spg_sym_array) {
type = SPG_YIELD_ARRAY;
} else if (pg_type == spg_sym_hash && rb_type(pg_value) == T_ARRAY) {
VALUE pg_value_key, pg_value_value;
pg_value_key = rb_ary_entry(pg_value, 0);
pg_value_value = rb_ary_entry(pg_value, 1);
if (SYMBOL_P(pg_value_key)) {
if (SYMBOL_P(pg_value_value)) {
type = SPG_YIELD_KV_HASH;
} else if (rb_type(pg_value_value) == T_ARRAY) {
type = SPG_YIELD_KMV_HASH;
}
} else if (rb_type(pg_value_key) == T_ARRAY) {
if (SYMBOL_P(pg_value_value)) {
type = SPG_YIELD_MKV_HASH;
} else if (rb_type(pg_value_value) == T_ARRAY) {
type = SPG_YIELD_MKMV_HASH;
}
}
} else if (pg_type == spg_sym_model && rb_type(pg_value) == T_CLASS) {
type = SPG_YIELD_MODEL;
}
}
}
switch(type) {
case SPG_YIELD_NORMAL:
/* Normal, hash for entire row */
for(i=0; i<ntuples; i++) {
h = rb_hash_new();
for(j=0; j<nfields; j++) {
rb_hash_aset(h, colsyms[j], spg__col_value(self, res, i, j, colconvert ENC_INDEX));
}
rb_yield(h);
}
break;
case SPG_YIELD_COLUMN:
/* Single column */
j = spg__field_id(pg_value, colsyms, nfields);
if (j == -1) {
for(i=0; i<ntuples; i++) {
rb_yield(Qnil);
}
} else {
for(i=0; i<ntuples; i++) {
rb_yield(spg__col_value(self, res, i, j, colconvert ENC_INDEX));
}
}
break;
case SPG_YIELD_COLUMNS:
/* Multiple columns as an array */
h = spg__field_ids(pg_value, colsyms, nfields);
for(i=0; i<ntuples; i++) {
rb_yield(spg__col_values(self, h, colsyms, nfields, res, i, colconvert ENC_INDEX));
}
break;
case SPG_YIELD_FIRST:
/* First column */
for(i=0; i<ntuples; i++) {
rb_yield(spg__col_value(self, res, i, 0, colconvert ENC_INDEX));
}
break;
case SPG_YIELD_ARRAY:
/* Array of all columns */
for(i=0; i<ntuples; i++) {
h = rb_ary_new2(nfields);
for(j=0; j<nfields; j++) {
rb_ary_store(h, j, spg__col_value(self, res, i, j, colconvert ENC_INDEX));
}
rb_yield(h);
}
break;
case SPG_YIELD_KV_HASH:
/* Hash with single key and single value */
{
VALUE k, v;
h = rb_hash_new();
k = spg__field_id(rb_ary_entry(pg_value, 0), colsyms, nfields);
v = spg__field_id(rb_ary_entry(pg_value, 1), colsyms, nfields);
for(i=0; i<ntuples; i++) {
rb_hash_aset(h, spg__col_value(self, res, i, k, colconvert ENC_INDEX), spg__col_value(self, res, i, v, colconvert ENC_INDEX));
}
rb_yield(h);
}
break;
case SPG_YIELD_MKV_HASH:
/* Hash with array of keys and single value */
{
VALUE k, v;
h = rb_hash_new();
k = spg__field_ids(rb_ary_entry(pg_value, 0), colsyms, nfields);
v = spg__field_id(rb_ary_entry(pg_value, 1), colsyms, nfields);
for(i=0; i<ntuples; i++) {
rb_hash_aset(h, spg__col_values(self, k, colsyms, nfields, res, i, colconvert ENC_INDEX), spg__col_value(self, res, i, v, colconvert ENC_INDEX));
}
rb_yield(h);
}
break;
case SPG_YIELD_KMV_HASH:
/* Hash with single keys and array of values */
{
VALUE k, v;
h = rb_hash_new();
k = spg__field_id(rb_ary_entry(pg_value, 0), colsyms, nfields);
v = spg__field_ids(rb_ary_entry(pg_value, 1), colsyms, nfields);
for(i=0; i<ntuples; i++) {
rb_hash_aset(h, spg__col_value(self, res, i, k, colconvert ENC_INDEX), spg__col_values(self, v, colsyms, nfields, res, i, colconvert ENC_INDEX));
}
rb_yield(h);
}
break;
case SPG_YIELD_MKMV_HASH:
/* Hash with array of keys and array of values */
{
VALUE k, v;
h = rb_hash_new();
k = spg__field_ids(rb_ary_entry(pg_value, 0), colsyms, nfields);
v = spg__field_ids(rb_ary_entry(pg_value, 1), colsyms, nfields);
for(i=0; i<ntuples; i++) {
rb_hash_aset(h, spg__col_values(self, k, colsyms, nfields, res, i, colconvert ENC_INDEX), spg__col_values(self, v, colsyms, nfields, res, i, colconvert ENC_INDEX));
}
rb_yield(h);
}
break;
case SPG_YIELD_MODEL:
/* Model object for entire row */
for(i=0; i<ntuples; i++) {
h = rb_hash_new();
for(j=0; j<nfields; j++) {
rb_hash_aset(h, colsyms[j], spg__col_value(self, res, i, j, colconvert ENC_INDEX));
}
/* Abuse local variable */
pg_type = rb_obj_alloc(pg_value);
rb_ivar_set(pg_type, spg_id_values, h);
rb_yield(pg_type);
}
break;
}
return self;
}
void Init_sequel_pg(void) {
VALUE c, spg_Postgres;
ID cg;
cg = rb_intern("const_get");
spg_id_new = rb_intern("new");
spg_id_local = rb_intern("local");
spg_id_year = rb_intern("year");
spg_id_month = rb_intern("month");
spg_id_day = rb_intern("day");
spg_id_output_identifier = rb_intern("output_identifier");
spg_id_datetime_class = rb_intern("datetime_class");
spg_id_application_timezone = rb_intern("application_timezone");
spg_id_timezone = rb_intern("timezone");
spg_id_op_plus = rb_intern("+");
spg_id_utc = rb_intern("utc");
spg_id_utc_offset = rb_intern("utc_offset");
spg_id_localtime = rb_intern("localtime");
spg_id_new_offset = rb_intern("new_offset");
spg_id_call = rb_intern("call");
spg_id_get = rb_intern("[]");
spg_id_opts = rb_intern("opts");
spg_id_db = rb_intern("db");
spg_id_conversion_procs = rb_intern("conversion_procs");
spg_id_columns = rb_intern("@columns");
spg_id_encoding = rb_intern("@encoding");
spg_id_values = rb_intern("@values");
spg_sym_utc = ID2SYM(rb_intern("utc"));
spg_sym_local = ID2SYM(rb_intern("local"));
spg_sym_map = ID2SYM(rb_intern("map"));
spg_sym_first = ID2SYM(rb_intern("first"));
spg_sym_array = ID2SYM(rb_intern("array"));
spg_sym_hash = ID2SYM(rb_intern("hash"));
spg_sym_model = ID2SYM(rb_intern("model"));
spg_sym__sequel_pg_type = ID2SYM(rb_intern("_sequel_pg_type"));
spg_sym__sequel_pg_value = ID2SYM(rb_intern("_sequel_pg_value"));
spg_Sequel = rb_funcall(rb_cObject, cg, 1, rb_str_new2("Sequel"));
spg_Blob = rb_funcall(rb_funcall(spg_Sequel, cg, 1, rb_str_new2("SQL")), cg, 1, rb_str_new2("Blob"));
spg_SQLTime= rb_funcall(spg_Sequel, cg, 1, rb_str_new2("SQLTime"));
spg_BigDecimal = rb_funcall(rb_cObject, cg, 1, rb_str_new2("BigDecimal"));
spg_Date = rb_funcall(rb_cObject, cg, 1, rb_str_new2("Date"));
spg_Postgres = rb_funcall(spg_Sequel, cg, 1, rb_str_new2("Postgres"));
rb_global_variable(&spg_Sequel);
rb_global_variable(&spg_Blob);
rb_global_variable(&spg_BigDecimal);
rb_global_variable(&spg_Date);
c = rb_funcall(spg_Postgres, cg, 1, rb_str_new2("Dataset"));
rb_define_private_method(c, "yield_hash_rows", spg_yield_hash_rows, 2);
rb_define_private_method(c, "fetch_rows_set_cols", spg_fetch_rows_set_cols, 1);
rb_require("sequel_pg/sequel_pg");
}
Jump to Line
Something went wrong with that request. Please try again.