pg 0.18.0 → 1.0.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (66) hide show
  1. checksums.yaml +5 -5
  2. checksums.yaml.gz.sig +0 -0
  3. data/BSDL +2 -2
  4. data/ChangeLog +1221 -4
  5. data/History.rdoc +130 -0
  6. data/Manifest.txt +0 -18
  7. data/README-Windows.rdoc +15 -26
  8. data/README.rdoc +16 -10
  9. data/Rakefile +32 -23
  10. data/Rakefile.cross +56 -38
  11. data/ext/errorcodes.def +33 -0
  12. data/ext/errorcodes.txt +15 -1
  13. data/ext/extconf.rb +27 -35
  14. data/ext/gvl_wrappers.c +4 -0
  15. data/ext/gvl_wrappers.h +27 -39
  16. data/ext/pg.c +19 -51
  17. data/ext/pg.h +22 -79
  18. data/ext/pg_binary_decoder.c +3 -1
  19. data/ext/pg_binary_encoder.c +14 -12
  20. data/ext/pg_coder.c +31 -10
  21. data/ext/pg_connection.c +350 -263
  22. data/ext/pg_copy_coder.c +34 -4
  23. data/ext/pg_result.c +27 -25
  24. data/ext/pg_text_decoder.c +9 -10
  25. data/ext/pg_text_encoder.c +93 -73
  26. data/ext/pg_type_map.c +20 -13
  27. data/ext/pg_type_map_by_column.c +7 -7
  28. data/ext/pg_type_map_by_mri_type.c +2 -2
  29. data/ext/pg_type_map_in_ruby.c +4 -7
  30. data/ext/util.c +3 -3
  31. data/ext/util.h +1 -1
  32. data/lib/pg/basic_type_mapping.rb +69 -42
  33. data/lib/pg/connection.rb +89 -38
  34. data/lib/pg/result.rb +10 -5
  35. data/lib/pg/text_decoder.rb +12 -3
  36. data/lib/pg/text_encoder.rb +8 -0
  37. data/lib/pg.rb +18 -10
  38. data/spec/helpers.rb +9 -16
  39. data/spec/pg/basic_type_mapping_spec.rb +58 -4
  40. data/spec/pg/connection_spec.rb +477 -217
  41. data/spec/pg/result_spec.rb +14 -7
  42. data/spec/pg/type_map_by_class_spec.rb +2 -2
  43. data/spec/pg/type_map_by_mri_type_spec.rb +1 -1
  44. data/spec/pg/type_spec.rb +145 -33
  45. data/spec/pg_spec.rb +1 -1
  46. data.tar.gz.sig +0 -0
  47. metadata +67 -66
  48. metadata.gz.sig +0 -0
  49. data/sample/array_insert.rb +0 -20
  50. data/sample/async_api.rb +0 -106
  51. data/sample/async_copyto.rb +0 -39
  52. data/sample/async_mixed.rb +0 -56
  53. data/sample/check_conn.rb +0 -21
  54. data/sample/copyfrom.rb +0 -81
  55. data/sample/copyto.rb +0 -19
  56. data/sample/cursor.rb +0 -21
  57. data/sample/disk_usage_report.rb +0 -186
  58. data/sample/issue-119.rb +0 -94
  59. data/sample/losample.rb +0 -69
  60. data/sample/minimal-testcase.rb +0 -17
  61. data/sample/notify_wait.rb +0 -72
  62. data/sample/pg_statistics.rb +0 -294
  63. data/sample/replication_monitor.rb +0 -231
  64. data/sample/test_binary_values.rb +0 -33
  65. data/sample/wal_shipper.rb +0 -434
  66. data/sample/warehouse_partitions.rb +0 -320
@@ -1,17 +0,0 @@
1
- #!/usr/bin/env ruby
2
-
3
- require 'pg'
4
-
5
- conn = PG.connect( :dbname => 'test' )
6
- $stderr.puts '---',
7
- RUBY_DESCRIPTION,
8
- PG.version_string( true ),
9
- "Server version: #{conn.server_version}",
10
- "Client version: #{PG.respond_to?( :library_version ) ? PG.library_version : 'unknown'}",
11
- '---'
12
-
13
- result = conn.exec( "SELECT * from pg_stat_activity" )
14
-
15
- $stderr.puts %Q{Expected this to return: ["select * from pg_stat_activity"]}
16
- p result.field_values( 'current_query' )
17
-
@@ -1,72 +0,0 @@
1
- #!/usr/bin/env ruby
2
- #
3
- # Test script, demonstrating a non-poll notification for a table event.
4
- #
5
-
6
- BEGIN {
7
- require 'pathname'
8
- basedir = Pathname.new( __FILE__ ).expand_path.dirname.parent
9
- libdir = basedir + 'lib'
10
- $LOAD_PATH.unshift( libdir.to_s ) unless $LOAD_PATH.include?( libdir.to_s )
11
- }
12
-
13
- require 'pg'
14
-
15
- TRIGGER_TABLE = %{
16
- CREATE TABLE IF NOT EXISTS test ( message text );
17
- }
18
-
19
- TRIGGER_FUNCTION = %{
20
- CREATE OR REPLACE FUNCTION notify_test()
21
- RETURNS TRIGGER
22
- LANGUAGE plpgsql
23
- AS $$
24
- BEGIN
25
- NOTIFY woo;
26
- RETURN NULL;
27
- END
28
- $$
29
- }
30
-
31
- DROP_TRIGGER = %{
32
- DROP TRIGGER IF EXISTS notify_trigger ON test
33
- }
34
-
35
-
36
- TRIGGER = %{
37
- CREATE TRIGGER notify_trigger
38
- AFTER UPDATE OR INSERT OR DELETE
39
- ON test
40
- FOR EACH STATEMENT
41
- EXECUTE PROCEDURE notify_test();
42
- }
43
-
44
- conn = PG.connect( :dbname => 'test' )
45
-
46
- conn.exec( TRIGGER_TABLE )
47
- conn.exec( TRIGGER_FUNCTION )
48
- conn.exec( DROP_TRIGGER )
49
- conn.exec( TRIGGER )
50
-
51
- conn.exec( 'LISTEN woo' ) # register interest in the 'woo' event
52
-
53
- notifications = []
54
-
55
- puts "Now switch to a different term and run:",
56
- '',
57
- %{ psql test -c "insert into test values ('A message.')"},
58
- ''
59
-
60
- puts "Waiting up to 30 seconds for for an event!"
61
- conn.wait_for_notify( 30 ) do |notify, pid|
62
- notifications << [ pid, notify ]
63
- end
64
-
65
- if notifications.empty?
66
- puts "Awww, I didn't see any events."
67
- else
68
- puts "I got one from pid %d: %s" % notifications.first
69
- end
70
-
71
-
72
-
@@ -1,294 +0,0 @@
1
- #!/usr/bin/env ruby
2
- # vim: set noet nosta sw=4 ts=4 :
3
- #
4
- # PostgreSQL statistic gatherer.
5
- # Mahlon E. Smith <mahlon@martini.nu>
6
- #
7
- # Based on queries by Kenny Gorman.
8
- # http://www.kennygorman.com/wordpress/?page_id=491
9
- #
10
- # An example gnuplot input script is included in the __END__ block
11
- # of this script. Using it, you can feed the output this script
12
- # generates to gnuplot (after removing header lines) to generate
13
- # some nice performance charts.
14
- #
15
-
16
- begin
17
- require 'ostruct'
18
- require 'optparse'
19
- require 'etc'
20
- require 'pg'
21
-
22
- rescue LoadError # 1.8 support
23
- unless Object.const_defined?( :Gem )
24
- require 'rubygems'
25
- retry
26
- end
27
- raise
28
- end
29
-
30
-
31
- ### PostgreSQL Stats. Fetch information from pg_stat_* tables.
32
- ### Optionally run in a continuous loop, displaying deltas.
33
- ###
34
- class Stats
35
- VERSION = %q$Id: pg_statistics.rb,v 36ca5b412583 2012/04/17 23:32:25 mahlon $
36
-
37
- def initialize( opts )
38
- @opts = opts
39
- @db = PG.connect(
40
- :dbname => opts.database,
41
- :host => opts.host,
42
- :port => opts.port,
43
- :user => opts.user,
44
- :password => opts.pass,
45
- :sslmode => 'prefer'
46
- )
47
- @last = nil
48
- end
49
-
50
- ######
51
- public
52
- ######
53
-
54
- ### Primary loop. Gather statistics and generate deltas.
55
- ###
56
- def run
57
- run_count = 0
58
-
59
- loop do
60
- current_stat = self.get_stats
61
-
62
- # First run, store and continue
63
- #
64
- if @last.nil?
65
- @last = current_stat
66
- sleep @opts.interval
67
- next
68
- end
69
-
70
- # headers
71
- #
72
- if run_count == 0 || run_count % 50 == 0
73
- puts "%-20s%12s%12s%12s%12s%12s%12s%12s%12s%12s%12s%12s%12s%12s%12s" % %w[
74
- time commits rollbks blksrd blkshit bkends seqscan
75
- seqtprd idxscn idxtrd ins upd del locks activeq
76
- ]
77
- end
78
-
79
- # calculate deltas
80
- #
81
- delta = current_stat.inject({}) do |h, pair|
82
- stat, val = *pair
83
-
84
- if %w[ activeq locks bkends ].include?( stat )
85
- h[stat] = current_stat[stat].to_i
86
- else
87
- h[stat] = current_stat[stat].to_i - @last[stat].to_i
88
- end
89
-
90
- h
91
- end
92
- delta[ 'time' ] = Time.now.strftime('%F %T')
93
-
94
- # new values
95
- #
96
- puts "%-20s%12s%12s%12s%12s%12s%12s%12s%12s%12s%12s%12s%12s%12s%12s" % [
97
- delta['time'], delta['commits'], delta['rollbks'], delta['blksrd'],
98
- delta['blkshit'], delta['bkends'], delta['seqscan'],
99
- delta['seqtprd'], delta['idxscn'], delta['idxtrd'],
100
- delta['ins'], delta['upd'], delta['del'], delta['locks'], delta['activeq']
101
- ]
102
-
103
- @last = current_stat
104
- run_count += 1
105
- sleep @opts.interval
106
- end
107
- end
108
-
109
-
110
- ### Query the database for performance measurements. Returns a hash.
111
- ###
112
- def get_stats
113
- res = @db.exec %Q{
114
- SELECT
115
- MAX(stat_db.xact_commit) AS commits,
116
- MAX(stat_db.xact_rollback) AS rollbks,
117
- MAX(stat_db.blks_read) AS blksrd,
118
- MAX(stat_db.blks_hit) AS blkshit,
119
- MAX(stat_db.numbackends) AS bkends,
120
- SUM(stat_tables.seq_scan) AS seqscan,
121
- SUM(stat_tables.seq_tup_read) AS seqtprd,
122
- SUM(stat_tables.idx_scan) AS idxscn,
123
- SUM(stat_tables.idx_tup_fetch) AS idxtrd,
124
- SUM(stat_tables.n_tup_ins) AS ins,
125
- SUM(stat_tables.n_tup_upd) AS upd,
126
- SUM(stat_tables.n_tup_del) AS del,
127
- MAX(stat_locks.locks) AS locks,
128
- MAX(activity.sess) AS activeq
129
- FROM
130
- pg_stat_database AS stat_db,
131
- pg_stat_user_tables AS stat_tables,
132
- (SELECT COUNT(*) AS locks FROM pg_locks ) AS stat_locks,
133
- (SELECT COUNT(*) AS sess FROM pg_stat_activity WHERE current_query <> '<IDLE>') AS activity
134
- WHERE
135
- stat_db.datname = '%s';
136
- } % [ @opts.database ]
137
-
138
- return res[0]
139
- end
140
- end
141
-
142
-
143
- ### Parse command line arguments. Return a struct of global options.
144
- ###
145
- def parse_args( args )
146
- options = OpenStruct.new
147
- options.database = Etc.getpwuid( Process.uid ).name
148
- options.host = '127.0.0.1'
149
- options.port = 5432
150
- options.user = Etc.getpwuid( Process.uid ).name
151
- options.sslmode = 'disable'
152
- options.interval = 5
153
-
154
- opts = OptionParser.new do |opts|
155
- opts.banner = "Usage: #{$0} [options]"
156
-
157
- opts.separator ''
158
- opts.separator 'Connection options:'
159
-
160
- opts.on( '-d', '--database DBNAME',
161
- "specify the database to connect to (default: \"#{options.database}\")" ) do |db|
162
- options.database = db
163
- end
164
-
165
- opts.on( '-h', '--host HOSTNAME', 'database server host' ) do |host|
166
- options.host = host
167
- end
168
-
169
- opts.on( '-p', '--port PORT', Integer,
170
- "database server port (default: \"#{options.port}\")" ) do |port|
171
- options.port = port
172
- end
173
-
174
- opts.on( '-U', '--user NAME',
175
- "database user name (default: \"#{options.user}\")" ) do |user|
176
- options.user = user
177
- end
178
-
179
- opts.on( '-W', 'force password prompt' ) do |pw|
180
- print 'Password: '
181
- begin
182
- system 'stty -echo'
183
- options.pass = gets.chomp
184
- ensure
185
- system 'stty echo'
186
- puts
187
- end
188
- end
189
-
190
- opts.separator ''
191
- opts.separator 'Other options:'
192
-
193
- opts.on( '-i', '--interval SECONDS', Integer,
194
- "refresh interval in seconds (default: \"#{options.interval}\")") do |seconds|
195
- options.interval = seconds
196
- end
197
-
198
- opts.on_tail( '--help', 'show this help, then exit' ) do
199
- $stderr.puts opts
200
- exit
201
- end
202
-
203
- opts.on_tail( '--version', 'output version information, then exit' ) do
204
- puts Stats::VERSION
205
- exit
206
- end
207
- end
208
-
209
- opts.parse!( args )
210
- return options
211
- end
212
-
213
-
214
- ### Go!
215
- ###
216
- if __FILE__ == $0
217
- $stdout.sync = true
218
- Stats.new( parse_args( ARGV ) ).run
219
- end
220
-
221
-
222
- __END__
223
- ######################################################################
224
- ### T E R M I N A L O P T I O N S
225
- ######################################################################
226
-
227
- #set terminal png nocrop enhanced font arial 8 size '800x600' x000000 xffffff x444444
228
- #set output 'graph.png'
229
-
230
- set terminal pdf linewidth 4 size 11,8
231
- set output 'graph.pdf'
232
-
233
- #set terminal aqua
234
-
235
-
236
- ######################################################################
237
- ### O P T I O N S F O R A L L G R A P H S
238
- ######################################################################
239
-
240
- set multiplot layout 2,1 title "PostgreSQL Statistics\n5 second sample rate (smoothed)"
241
-
242
- set grid x y
243
- set key right vertical outside
244
- set key nobox
245
- set xdata time
246
- set timefmt "%Y-%m-%d.%H:%M:%S"
247
- set format x "%l%p"
248
- set xtic rotate by -45
249
- input_file = "database_stats.txt"
250
-
251
- # edit to taste!
252
- set xrange ["2012-04-16.00:00:00":"2012-04-17.00:00:00"]
253
-
254
-
255
- ######################################################################
256
- ### G R A P H 1
257
- ######################################################################
258
-
259
- set title "Database Operations and Connection Totals"
260
- set yrange [0:200]
261
-
262
- plot \
263
- input_file using 1:2 title "Commits" with lines smooth bezier, \
264
- input_file using 1:3 title "Rollbacks" with lines smooth bezier, \
265
- input_file using 1:11 title "Inserts" with lines smooth bezier, \
266
- input_file using 1:12 title "Updates" with lines smooth bezier, \
267
- input_file using 1:13 title "Deletes" with lines smooth bezier, \
268
- input_file using 1:6 title "Backends (total)" with lines, \
269
- input_file using 1:15 title "Active queries (total)" with lines smooth bezier
270
-
271
-
272
- ######################################################################
273
- ### G R A P H 2
274
- ######################################################################
275
-
276
- set title "Backend Performance"
277
- set yrange [0:10000]
278
-
279
- plot \
280
- input_file using 1:4 title "Block (cache) reads" with lines smooth bezier, \
281
- input_file using 1:5 title "Block (cache) hits" with lines smooth bezier, \
282
- input_file using 1:7 title "Sequence scans" with lines smooth bezier, \
283
- input_file using 1:8 title "Sequence tuple reads" with lines smooth bezier, \
284
- input_file using 1:9 title "Index scans" with lines smooth bezier, \
285
- input_file using 1:10 title "Index tuple reads" with lines smooth bezier
286
-
287
-
288
- ######################################################################
289
- ### C L E A N U P
290
- ######################################################################
291
-
292
- unset multiplot
293
- reset
294
-
@@ -1,231 +0,0 @@
1
- #!/usr/bin/env ruby
2
- # vim: set noet nosta sw=4 ts=4 :
3
- #
4
- # Get the current WAL segment and offset from a master postgresql
5
- # server, and compare slave servers to see how far behind they
6
- # are in MB. This script should be easily modified for use with
7
- # Nagios/Mon/Monit/Zabbix/whatever, or wrapping it in a display loop,
8
- # and is suitable for both WAL shipping or streaming forms of replication.
9
- #
10
- # Mahlon E. Smith <mahlon@martini.nu>
11
- #
12
- # First argument is the master server, all other arguments are treated
13
- # as slave machines.
14
- #
15
- # db_replication.monitor db-master.example.com ...
16
- #
17
-
18
- begin
19
- require 'ostruct'
20
- require 'optparse'
21
- require 'pathname'
22
- require 'etc'
23
- require 'pg'
24
- require 'pp'
25
-
26
- rescue LoadError # 1.8 support
27
- unless Object.const_defined?( :Gem )
28
- require 'rubygems'
29
- retry
30
- end
31
- raise
32
- end
33
-
34
-
35
- ### A class to encapsulate the PG handles.
36
- ###
37
- class PGMonitor
38
-
39
- VERSION = %q$Id: replication_monitor.rb,v 36ca5b412583 2012/04/17 23:32:25 mahlon $
40
-
41
- # When to consider a slave as 'behind', measured in WAL segments.
42
- # The default WAL segment size is 16, so we'll alert after
43
- # missing two WAL files worth of data.
44
- #
45
- LAG_ALERT = 32
46
-
47
- ### Create a new PGMonitor object.
48
- ###
49
- def initialize( opts, hosts )
50
- @opts = opts
51
- @master = hosts.shift
52
- @slaves = hosts
53
- @current_wal = {}
54
- @failures = []
55
- end
56
-
57
- attr_reader :opts, :current_wal, :master, :slaves, :failures
58
-
59
-
60
- ### Perform the connections and check the lag.
61
- ###
62
- def check
63
- # clear prior failures, get current xlog info
64
- @failures = []
65
- return unless self.get_current_wal
66
-
67
- # check all slaves
68
- self.slaves.each do |slave|
69
- begin
70
- slave_db = PG.connect(
71
- :dbname => self.opts.database,
72
- :host => slave,
73
- :port => self.opts.port,
74
- :user => self.opts.user,
75
- :password => self.opts.pass,
76
- :sslmode => 'prefer'
77
- )
78
-
79
- xlog = slave_db.exec( 'SELECT pg_last_xlog_receive_location()' ).getvalue( 0, 0 )
80
- slave_db.close
81
-
82
- lag_in_megs = ( self.find_lag( xlog ).to_f / 1024 / 1024 ).abs
83
- if lag_in_megs >= LAG_ALERT
84
- failures << { :host => slave,
85
- :error => "%0.2fMB behind the master." % [ lag_in_megs ] }
86
- end
87
- rescue => err
88
- failures << { :host => slave, :error => err.message }
89
- end
90
- end
91
- end
92
-
93
-
94
- #########
95
- protected
96
- #########
97
-
98
- ### Ask the master for the current xlog information, to compare
99
- ### to slaves. Returns true on succcess. On failure, populates
100
- ### the failures array and returns false.
101
- ###
102
- def get_current_wal
103
- master_db = PG.connect(
104
- :dbname => self.opts.database,
105
- :host => self.master,
106
- :port => self.opts.port,
107
- :user => self.opts.user,
108
- :password => self.opts.pass,
109
- :sslmode => 'prefer'
110
- )
111
-
112
- self.current_wal[ :segbytes ] = master_db.exec( 'SHOW wal_segment_size' ).
113
- getvalue( 0, 0 ).sub( /\D+/, '' ).to_i << 20
114
-
115
- current = master_db.exec( 'SELECT pg_current_xlog_location()' ).getvalue( 0, 0 )
116
- self.current_wal[ :segment ], self.current_wal[ :offset ] = current.split( /\// )
117
-
118
- master_db.close
119
- return true
120
-
121
- # If we can't get any of the info from the master, then there is no
122
- # point in a comparison with slaves.
123
- #
124
- rescue => err
125
- self.failures << { :host => self.master,
126
- :error => 'Unable to retrieve required info from the master (%s)' % [ err.message ] }
127
-
128
- return false
129
- end
130
-
131
-
132
- ### Given an +xlog+ position from a slave server, return
133
- ### the number of bytes the slave needs to replay before it
134
- ### is caught up to the master.
135
- ###
136
- def find_lag( xlog )
137
- s_segment, s_offset = xlog.split( /\// )
138
- m_segment = self.current_wal[ :segment ]
139
- m_offset = self.current_wal[ :offset ]
140
- m_segbytes = self.current_wal[ :segbytes ]
141
-
142
- return (( m_segment.hex - s_segment.hex ) * m_segbytes) + ( m_offset.hex - s_offset.hex )
143
- end
144
-
145
- end
146
-
147
-
148
- ### Parse command line arguments. Return a struct of global options.
149
- ###
150
- def parse_args( args )
151
- options = OpenStruct.new
152
- options.database = 'postgres'
153
- options.port = 5432
154
- options.user = Etc.getpwuid( Process.uid ).name
155
- options.sslmode = 'prefer'
156
-
157
- opts = OptionParser.new do |opts|
158
- opts.banner = "Usage: #{$0} [options] <master> <slave> [slave2, slave3...]"
159
-
160
- opts.separator ''
161
- opts.separator 'Connection options:'
162
-
163
- opts.on( '-d', '--database DBNAME',
164
- "specify the database to connect to (default: \"#{options.database}\")" ) do |db|
165
- options.database = db
166
- end
167
-
168
- opts.on( '-h', '--host HOSTNAME', 'database server host' ) do |host|
169
- options.host = host
170
- end
171
-
172
- opts.on( '-p', '--port PORT', Integer,
173
- "database server port (default: \"#{options.port}\")" ) do |port|
174
- options.port = port
175
- end
176
-
177
- opts.on( '-U', '--user NAME',
178
- "database user name (default: \"#{options.user}\")" ) do |user|
179
- options.user = user
180
- end
181
-
182
- opts.on( '-W', 'force password prompt' ) do |pw|
183
- print 'Password: '
184
- begin
185
- system 'stty -echo'
186
- options.pass = $stdin.gets.chomp
187
- ensure
188
- system 'stty echo'
189
- puts
190
- end
191
- end
192
-
193
- opts.separator ''
194
- opts.separator 'Other options:'
195
-
196
- opts.on_tail( '--help', 'show this help, then exit' ) do
197
- $stderr.puts opts
198
- exit
199
- end
200
-
201
- opts.on_tail( '--version', 'output version information, then exit' ) do
202
- puts PGMonitor::VERSION
203
- exit
204
- end
205
- end
206
-
207
- opts.parse!( args )
208
- return options
209
- end
210
-
211
-
212
-
213
- if __FILE__ == $0
214
- opts = parse_args( ARGV )
215
- raise ArgumentError, "At least two PostgreSQL servers are required." if ARGV.length < 2
216
- mon = PGMonitor.new( opts, ARGV )
217
-
218
- mon.check
219
- if mon.failures.empty?
220
- puts "All is well!"
221
- exit 0
222
- else
223
- puts "Database replication delayed or broken."
224
- mon.failures.each do |bad|
225
- puts "%s: %s" % [ bad[ :host ], bad[ :error ] ]
226
- end
227
- exit 1
228
- end
229
- end
230
-
231
-
@@ -1,33 +0,0 @@
1
- #!/usr/bin/env ruby1.9.1
2
-
3
- require 'pg'
4
-
5
- db = PG.connect( :dbname => 'test' )
6
- db.exec "DROP TABLE IF EXISTS test"
7
- db.exec "CREATE TABLE test (a INTEGER, b BYTEA)"
8
-
9
- a = 42
10
- b = [1, 2, 3]
11
- db.exec "INSERT INTO test(a, b) VALUES($1::int, $2::bytea)",
12
- [a, {:value => b.pack('N*'), :format => 1}]
13
-
14
- db.exec( "SELECT a::int, b::bytea FROM test LIMIT 1", [], 1 ) do |res|
15
-
16
- res.nfields.times do |i|
17
- puts "Field %d is: %s, a %s (%s) column from table %p" % [
18
- i,
19
- res.fname( i ),
20
- db.exec( "SELECT format_type($1,$2)", [res.ftype(i), res.fmod(1)] ).getvalue(0,0),
21
- res.fformat( i ).zero? ? "string" : "binary",
22
- res.ftable( i ),
23
- ]
24
- end
25
-
26
- res.each do |row|
27
- puts "a = #{row['a'].inspect}"
28
- puts "a (unpacked) = #{row['a'].unpack('N*').inspect}"
29
- puts "b = #{row['b'].unpack('N*').inspect}"
30
- end
31
- end
32
-
33
-