pillboxr 0.0.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
data/.gitignore ADDED
@@ -0,0 +1,4 @@
1
+ test_api_key.yml
2
+ doc
3
+ *.gem
4
+ .bundle
data/Gemfile ADDED
@@ -0,0 +1,4 @@
1
+ source :gemcutter
2
+
3
+ # Specify your gem's dependencies in pillbox.gemspec
4
+ gemspec
data/Gemfile.lock ADDED
@@ -0,0 +1,27 @@
1
+ PATH
2
+ remote: .
3
+ specs:
4
+ pillboxr (0.0.1)
5
+ activeresource (> 2.3.5)
6
+
7
+ GEM
8
+ remote: http://rubygems.org/
9
+ specs:
10
+ activemodel (3.0.0)
11
+ activesupport (= 3.0.0)
12
+ builder (~> 2.1.2)
13
+ i18n (~> 0.4.1)
14
+ activeresource (3.0.0)
15
+ activemodel (= 3.0.0)
16
+ activesupport (= 3.0.0)
17
+ activesupport (3.0.0)
18
+ builder (2.1.2)
19
+ i18n (0.4.1)
20
+
21
+ PLATFORMS
22
+ ruby
23
+
24
+ DEPENDENCIES
25
+ activeresource (> 2.3.5)
26
+ bundler (>= 1.0.0)
27
+ pillboxr!
data/README.rdoc ADDED
@@ -0,0 +1,44 @@
1
+ = Pillboxr
2
+
3
+ Pillboxr is a Ruby wrapper for the National Library of Medicine Pillbox API Service located at http://pillbox.nlm.nih.gov/PHP/pillboxAPIService.php.
4
+
5
+ The pillbox API provides information from the FDA about various prescription medications.
6
+
7
+ Pillboxr inherits from ActiveResource to perform it's XML wrapping so ActiveResource 3.* or 2.3.* is a requirement for using the wrapper. Please see the doc directory for documentation on using the library.
8
+
9
+ = Usage
10
+
11
+ Getting started is fairly easy:
12
+
13
+ >> require 'pillboxr' # You may have to require rubygems first
14
+
15
+ >> Pillboxr.api_key = 'YOUR API KEY HERE' # See below for directions on obtaining an API key
16
+
17
+ >> Pillboxr.all(:params => {:has_image => 1}) # Find all pills in the database with images associated
18
+
19
+ The params hash may include any of the following parameters:
20
+
21
+ 'color' => 'string' or Array with multiple colors (see http://pillbox.nlm.nih.gov/API-documentation.html)
22
+ 'score' => integer
23
+ 'ingredient' => 'string' or Array with multiple ingredients (returned results include all ingredients)
24
+ 'inactive' => 'string'
25
+ 'dea' => 'string' or any of 'I, II, III, IV, V'
26
+ 'author' => 'string'
27
+ 'shape' => 'string' (Shape or Hex)
28
+ 'imprint' => 'string'
29
+ 'prodcode' => 'string' (Product Code: see http://pillbox.nlm.nih.gov/API-documentation.html)
30
+ 'has_image' => 0 or 1 for no image present or image present respectively
31
+ 'size' => integer for size in millimeters (currently this encompasses a range of +/- 2 mm)
32
+ 'lower_limit' => integer for which returned record to start at (currently non-functional)
33
+
34
+ Please see specific files or the document directory for specific usage examples. Further API documentation available on project homepage: http://pillbox.nlm.nih.gov/API-documentation.html
35
+
36
+ = KNOWN BUGS
37
+
38
+ * Please note that some XML in the Pillbox API is unescaped.
39
+
40
+ API provided through the generous support by the FDA in both money and resources. Work conducted by NLM at NIH.
41
+
42
+ Please contact david.hale at nlm.nih.gov for an api key. There is no bandwidth limit currently.
43
+
44
+ Data is owned by companies, mandatorily licenced for X purposes.
data/Rakefile ADDED
@@ -0,0 +1,15 @@
1
+ require 'rubygems'
2
+ require 'rake'
3
+ require 'rake/testtask'
4
+ require 'bundler'
5
+
6
+ Bundler::GemHelper.install_tasks
7
+
8
+ desc "Run all tests for this project."
9
+ Rake::TestTask.new(:test) do |test|
10
+ test.libs << 'lib' << 'test'
11
+ test.pattern = 'test/**/*_test.rb'
12
+ test.verbose = true
13
+ end
14
+
15
+ task :default => :test
@@ -0,0 +1,27 @@
1
+ ##
2
+ # Pillboxr monkeypatches the ActiveResource::Base.instantiate_collection method
3
+ # to remove the disclaimer included in all XML returned from the Pillbox API Service
4
+ module ActiveResource
5
+ class Base
6
+ def self.instantiate_collection(collection, prefix_options = {}) # :nodoc:
7
+ if collection.is_a?(Hash) && collection.size == 1
8
+ value = collection.values.first
9
+ if value.is_a?(Array)
10
+ value.collect! { |record| instantiate_record(record, prefix_options) }
11
+ else
12
+ [ instantiate_record(value, prefix_options) ]
13
+ end
14
+ else
15
+ # strip extra layer off the front end (a disclaimer)
16
+ (d,disclaimer), (p,collection), (r,@record_count) = collection.sort
17
+
18
+ puts "\nMatched #{@record_count} records...\n"
19
+
20
+ # ensure array
21
+ collection = collection.is_a?(Array) ? collection : Array[collection]
22
+
23
+ collection.collect! { |record| instantiate_record(record, prefix_options) }
24
+ end
25
+ end
26
+ end
27
+ end
@@ -0,0 +1,15 @@
1
+ module Compatibility
2
+ module ClassMethods
3
+ def first(options = {})
4
+ find(:first, options)
5
+ end
6
+
7
+ def all(options = {})
8
+ find(:all, options)
9
+ end
10
+ end
11
+
12
+ def self.included(base)
13
+ base.extend(ClassMethods)
14
+ end
15
+ end
@@ -0,0 +1,40 @@
1
+ require 'nokogiri'
2
+ require 'open-uri'
3
+ class PillboxResourceNoko
4
+ attr_accessor :attrs
5
+ def PillboxResourceNoko::find_first_with_img_by_ingredient(ingredient)
6
+ prs = PillboxResourceNoko.find_all_by_ingredient(ingredient)
7
+ pill = nil
8
+ prs.each do |pr|
9
+ pill = pr if pr.has_image?
10
+ end
11
+ pill
12
+ end
13
+ def PillboxResourceNoko::find_all_by_ingredient(ingredient)
14
+ doc = Nokogiri::XML(open("http://pillbox.nlm.nih.gov/PHP/pillboxAPIService.php?key=12345&ingredient=#{ingredient}"))
15
+ prs = []
16
+ doc.xpath('//Pills/pill').each do |pill|
17
+ pr = PillboxResourceNoko.new
18
+ pr.attrs = {}
19
+ pill.children.each do |child|
20
+ next if child.nil? or child.name == "text"
21
+ pr.attrs[child.name] = child.inner_html
22
+ end
23
+ prs << pr
24
+ end
25
+ prs
26
+ end
27
+
28
+ def shape; attrs['SPLSHAPE'] end
29
+ def color; attrs['SPLCOLOR'] end
30
+
31
+ def description; attrs['RXSTRING'] end
32
+ def product_code; attrs['PRODUCT_CODE'] end
33
+ def has_image?; attrs['HAS_IMAGE'] == '1' end
34
+ def ingredients; attrs['INGREDIENTS'].split(";") end
35
+ def size; attrs['SPLSIZE'].to_i end
36
+ def image_id; attrs['image_id'] end
37
+ def image_url; image_id ? "http://pillbox.nlm.nih.gov/assets/super_small/#{image_id}ss.png" : nil end
38
+ #def image_url; image_id ? "http://pillbox.nlm.nih.gov/assets/small/#{image_id}sm.jpg" : nil end
39
+ def imprint; attrs['splimprint'] end
40
+ end
data/lib/pillboxr.rb ADDED
@@ -0,0 +1,552 @@
1
+ # -*- encoding: utf-8 -*-
2
+ $:.unshift(File.dirname(__FILE__))
3
+
4
+ begin
5
+ require 'active_resource'
6
+ rescue LoadError
7
+ begin
8
+ require 'rubygems'
9
+ require 'active_resource'
10
+ rescue LoadError
11
+ abort <<-ERROR
12
+ The 'activeresource' or 'pillboxr' library could not be loaded. If you have RubyGems
13
+ installed you can install ActiveResource by doing "gem install activeresource". If ActiveResource is installed
14
+ you may need to change your path to allow Pillboxr to load.
15
+ ERROR
16
+ end
17
+ end
18
+
19
+ require 'compatibility'
20
+ require 'active_resource/version'
21
+ require 'activeresource_patch'
22
+ =begin rdoc
23
+
24
+ NOTE: This library utilizes version 2 of the Pillbox API, published 10/9/10.
25
+
26
+ USAGE
27
+ require 'pillboxr'
28
+
29
+ name = 'aspirin'
30
+
31
+ Pillboxr.api_key = "YOUR SECRET KEY"
32
+ pills = Pillboxr.find(:all, :params=>{"ingredient"=>name})
33
+ if pills.empty?
34
+ puts "could not find #{name}"
35
+ else
36
+ ...
37
+ end
38
+ =end
39
+
40
+ ##
41
+ # Pillboxr is a subclass of ActiveResource::Base that provides additional convenience
42
+ # methods and some parameter wrapping for querying the Pillbox API Service located at
43
+ # http://pillbox.nlm.nih.gov/PHP/pillboxAPIService.php
44
+ class Pillboxr < ActiveResource::Base
45
+ ARES_VERSIONS = ['2.3.4', '2.3.5', '2.3.8', '2.3.9', '2.3.10', '3.0.0']
46
+
47
+ if ActiveResource::VERSION::STRING < '3.0.0'
48
+ include Compatibility
49
+ end
50
+
51
+ # Version Check
52
+ unless ARES_VERSIONS.include?(ActiveResource::VERSION::STRING)
53
+ abort <<-ERROR
54
+ ActiveResource version #{ARES_VERSIONS.join(' or ')} is required.
55
+ ERROR
56
+ end
57
+
58
+ self.site = "http://pillbox.nlm.nih.gov/PHP/pillboxAPIService.php"
59
+
60
+ SHAPES = {
61
+ 'BULLET'=> 'C48335',
62
+ 'CAPSULE'=> 'C48336',
63
+ 'CLOVER'=> 'C48337',
64
+ 'DIAMOND'=> 'C48338',
65
+ 'DOUBLE_CIRCLE'=> 'C48339',
66
+ 'FREEFORM'=> 'C48340',
67
+ 'GEAR'=> 'C48341',
68
+ 'HEPTAGON'=> 'C48342',
69
+ 'HEXAGON'=> 'C48343',
70
+ 'OCTAGON'=> 'C48344',
71
+ 'OVAL'=> 'C48345',
72
+ 'PENTAGON'=> 'C48346',
73
+ 'RECTANGLE'=> 'C48347',
74
+ 'ROUND'=> 'C48348',
75
+ 'SEMI_CIRCLE'=> 'C48349',
76
+ 'SQUARE'=> 'C48350',
77
+ 'TEAR'=> 'C48351',
78
+ 'TRAPEZOID'=> 'C48352',
79
+ 'TRIANGLE'=> 'C48353'
80
+ }
81
+ SHAPE_CODES = SHAPES.invert
82
+
83
+ ##
84
+ # Returns a hash of the accepted pill shapes and their related codes.
85
+ # {
86
+ # 'BULLET'=> 'C48335',
87
+ # 'CAPSULE'=> 'C48336',
88
+ # 'CLOVER'=> 'C48337',
89
+ # 'DIAMOND'=> 'C48338',
90
+ # 'DOUBLE_CIRCLE'=> 'C48339',
91
+ # 'FREEFORM'=> 'C48340',
92
+ # 'GEAR'=> 'C48341',
93
+ # 'HEPTAGON'=> 'C48342',
94
+ # 'HEXAGON'=> 'C48343',
95
+ # 'OCTAGON'=> 'C48344',
96
+ # 'OVAL'=> 'C48345',
97
+ # 'PENTAGON'=> 'C48346',
98
+ # 'RECTANGLE'=> 'C48347',
99
+ # 'ROUND'=> 'C48348',
100
+ # 'SEMI_CIRCLE'=> 'C48349',
101
+ # 'SQUARE'=> 'C48350',
102
+ # 'TEAR'=> 'C48351',
103
+ # 'TRAPEZOID'=> 'C48352',
104
+ # 'TRIANGLE'=> 'C48353'
105
+ # }
106
+ def self.shapes
107
+ SHAPES.inject({}){|i,(k,v)| i.merge k.humanize => v }
108
+ end
109
+
110
+ COLORS = {
111
+ 'BLACK'=> 'C48323',
112
+ 'BLUE'=> 'C48333',
113
+ 'BROWN'=> 'C48332',
114
+ 'GRAY'=> 'C48324',
115
+ 'GREEN'=> 'C48329',
116
+ 'ORANGE'=> 'C48331',
117
+ 'PINK'=> 'C48328',
118
+ 'PURPLE'=> 'C48327',
119
+ 'RED'=> 'C48326',
120
+ 'TURQUOISE'=> 'C48334',
121
+ 'WHITE'=> 'C48325',
122
+ 'YELLOW'=> 'C48330'
123
+ }
124
+ COLOR_CODES = COLORS.invert
125
+
126
+ ##
127
+ # Returns a hash of the accepted pill colors and their related codes.
128
+ # {
129
+ # 'BLACK'=> 'C48323',
130
+ # 'BLUE'=> 'C48333',
131
+ # 'BROWN'=> 'C48332',
132
+ # 'GRAY'=> 'C48324',
133
+ # 'GREEN'=> 'C48329',
134
+ # 'ORANGE'=> 'C48331',
135
+ # 'PINK'=> 'C48328',
136
+ # 'PURPLE'=> 'C48327',
137
+ # 'RED'=> 'C48326',
138
+ # 'TURQUOISE'=> 'C48334',
139
+ # 'WHITE'=> 'C48325',
140
+ # 'YELLOW'=> 'C48330'
141
+ # }
142
+ def self.colors
143
+ COLORS.inject({}){|i,(k,v)| i.merge k.humanize => v }
144
+ end
145
+
146
+ DEA_CODES = {
147
+ 'I' => 'C48672',
148
+ 'II' => 'C48675',
149
+ 'III' => 'C48676',
150
+ 'IV' => 'C48677',
151
+ 'V' => 'C48679'
152
+ }
153
+ SCHEDULE_CODES = DEA_CODES.invert
154
+
155
+ ##
156
+ # Returns a hash of the accepted DEA Schedule codes and their related API codes
157
+ # {
158
+ # 'I' => 'C48672',
159
+ # 'II' => 'C48675',
160
+ # 'III' => 'C48676',
161
+ # 'IV' => 'C48677',
162
+ # 'V' => 'C48679'
163
+ # }
164
+ def self.dea_codes
165
+ DEA_CODES.inject({}) { |i,(k,v)| i.merge k.humanize => v }
166
+ end
167
+
168
+ cattr_accessor :api_key
169
+
170
+ ##
171
+ # Returns as an integer the number of records returned from the previous API call.
172
+ def self.record_count
173
+ @record_count.to_i
174
+ end
175
+
176
+ ##
177
+ # This class method sets the api_key parameter to the default key listed in the yaml
178
+ # file under fixtures/test_api_key.yml. This is useful for putting a call to
179
+ # Pillboxr.test! in your setup method in your test suite
180
+ def self.test!
181
+ "Using testing api_key" if self.api_key = load_test_key
182
+ end
183
+
184
+ ##
185
+ # Accepts a first argument of :first or :all to narrow search results. Accepts a params
186
+ # hash as the second argument consistent with ActiveResource.find.
187
+ #
188
+ # Accepted parameters are:
189
+ # 'color' => 'string' or Array with multiple colors (see http://pillbox.nlm.nih.gov/API-documentation.html)
190
+ # 'score' => integer
191
+ # 'ingredient' => 'string' or Array with multiple ingredients (returned results include all ingredients)
192
+ # 'inactive' => 'string'
193
+ # 'dea' => 'string' or any of 'I, II, III, IV, V'
194
+ # 'author' => 'string'
195
+ # 'shape' => 'string' (Shape or Hex)
196
+ # 'imprint' => 'string'
197
+ # 'prodcode' => 'string' (Product Code: see http://pillbox.nlm.nih.gov/API-documentation.html)
198
+ # 'has_image' => 0 or 1 for no image present or image present respectively
199
+ # 'size' => integer for size in millimeters (currently this encompasses a range of +/- 2 mm)
200
+ # 'lower_limit' => integer for which returned record to start at (currently non-functional)
201
+ #
202
+ def self.find(first, options={})
203
+ # MYTODO :| ok for now... but this only works with rails
204
+ options = HashWithIndifferentAccess.new(options)
205
+ validate_pillbox_api_params(options)
206
+ begin
207
+ super first, self.interpret_params(options)
208
+ rescue REXML::ParseException => e
209
+ # Work around no XML returned when no records are found.
210
+ if e.message =~ /The document "No records found" does not have a valid root/
211
+ STDERR.puts "No records found."
212
+ else
213
+ raise
214
+ end
215
+ end
216
+ end
217
+
218
+ ##
219
+ # These two metaprogramming methods implemented to give flexibility when querying a Pillboxr object
220
+ # for its information.
221
+ def respond_to?(meth) # :nodoc:
222
+ (attributes.has_key?(meth.to_s.upcase) || attributes.has_key?(meth.to_s)) ? true : super
223
+ end
224
+
225
+ def method_missing(method, *args, &block) # :nodoc:
226
+ if attributes.has_key?(method.to_s.upcase)
227
+ attributes[method.to_s.upcase].nil? ? [] : attributes[method.to_s.upcase]
228
+ elsif attributes.has_key?(method.to_s)
229
+ attributes[method.to_s].nil? ? [] : attributes[method.to_s]
230
+ else
231
+ super
232
+ end
233
+ end
234
+
235
+ ##
236
+ # Returns the pill shape as a human readable shape description or as a 'shape code'.
237
+ def shape # handle multi-color
238
+ return nil unless attributes['SPLSHAPE']
239
+ attributes['SPLSHAPE'].split(";").map do |shape_code|
240
+ SHAPE_CODES[shape_code] || shape_code
241
+ end
242
+ end
243
+
244
+ ##
245
+ # Returns the pill color as a human readable color description or as a 'color code'.
246
+ def color
247
+ return nil unless attributes['SPLCOLOR']
248
+ attributes['SPLCOLOR'].split(";").map do |color_code|
249
+ COLOR_CODES[color_code] || color_code
250
+ end
251
+ end
252
+
253
+ ##
254
+ # Returns the product code as a string, 9-digit FDA code listed at:
255
+ # http://www.fda.gov/Drugs/InformationOnDrugs/ucm142438.htm
256
+ def prodcode; attributes['PRODUCT_CODE'] end
257
+
258
+ ##
259
+ # Returns a url as a string for looking up the drug monograph using the
260
+ # http://druginfo.nlm.nih.gov/drugportal API
261
+ def info_url; "http://druginfo.nlm.nih.gov/drugportal/dpdirect.jsp?name="+ingredient end
262
+
263
+ ##
264
+ # Returns true if the pill has an associated image, false if it does not.
265
+ def has_image?; attributes['HAS_IMAGE'] == '1' end
266
+
267
+ ##
268
+ # Returns a string list of ingredients. Does not contain brand/trade names.
269
+ def ingredient; self.ingredients end
270
+
271
+ ##
272
+ # Returns the size of the pill in millimeters as a float. Values contain two decimal places.
273
+ def size; attributes['SPLSIZE'].to_f end
274
+
275
+ ##
276
+ # Returns an integer representing the score value see http://goo.gl/SzCO for details.
277
+ def score; attributes['SPLSCORE'] end
278
+
279
+ ##
280
+ # Returns the string representation of any alphanumeric text appearing on the pill.
281
+ def imprint; attributes['splimprint'] end
282
+
283
+ ##
284
+ # Returns the trade brand/trade name as a capitalized string. This is not the generic name of the active ingredient.
285
+ # For example the trade name of 'sildenafil' would be 'Viagra'.
286
+ def trade_name; self.rxstring.split(" ").first.downcase.capitalize end
287
+
288
+ ##
289
+ # Accepts image_size argument as one of super_small, small, medium, large, or all. Default is super_small.
290
+ # Returns the url, as a string, of the corresponding image when all is not given as the image size argument.
291
+ # Returns an array of urls representing all sizes of the pill image when 'all' is given as an argument
292
+ def image_url(image_size = 'super_small')
293
+ unless image_id
294
+ return nil
295
+ end
296
+ case image_size
297
+ when "super_small"; "http://pillbox.nlm.nih.gov/assets/super_small/#{image_id}ss.png"
298
+ when "small"; "http://pillbox.nlm.nih.gov/assets/small/#{image_id}sm.jpg"
299
+ when "medium"; "http://pillbox.nlm.nih.gov/assets/medium/#{image_id}md.jpg"
300
+ when "large"; "http://pillbox.nlm.nih.gov/assets/large/#{image_id}lg.jpg"
301
+ when "all"
302
+ ["http://pillbox.nlm.nih.gov/assets/super_small/#{image_id}ss.png",
303
+ "http://pillbox.nlm.nih.gov/assets/small/#{image_id}sm.jpg",
304
+ "http://pillbox.nlm.nih.gov/assets/medium/#{image_id}md.jpg",
305
+ "http://pillbox.nlm.nih.gov/assets/large/#{image_id}lg.jpg"]
306
+ end
307
+ end
308
+
309
+ ##
310
+ # Returns a string with the DEA schedule code with 'Schedule' prepended, i.e...
311
+ # pill.dea => 'Schedule II'
312
+ def dea
313
+ return nil unless attributes['DEA_SCHEDULE_CODE']
314
+ "Schedule #{SCHEDULE_CODES[attributes['DEA_SCHEDULE_CODE'].to_s]}"
315
+ end
316
+
317
+ ##
318
+ # Returns the inactive ingredients as an array with double quotes removed. In the current API
319
+ # double quotes are used instead of spaces between ingredients.
320
+ def inactive
321
+ # Remove double quotes from the inactive ingredients list
322
+ Array(attributes['SPL_INACTIVE_ING'].split("/")).map { |str| str.gsub('"', ' ').strip}
323
+ end
324
+
325
+ ##
326
+ # Returns a string representing the author of the drug monograph with spaces and double quotes removed.
327
+ def author
328
+ # Remove double quotes from the author attribute
329
+ attributes['AUTHOR'].gsub('"', ' ').strip
330
+ end
331
+
332
+ private
333
+ def self.load_test_key # :nodoc:
334
+ test_key = YAML.load_file("#{File.expand_path(File.dirname(__FILE__) + '/../test/fixtures/test_api_key.yml')}")
335
+ test_key[:key]
336
+ end
337
+
338
+ VALID_ATTRIBUTE_NAMES = %w(color score ingredient ingredients inactive dea author shape imprint prodcode has_image size lower_limit key)
339
+
340
+ def self.validate_pillbox_api_params(options) # :nodoc:
341
+ validate_presence_of_api_key(options)
342
+ raise "try using find :all, :params => { ... } with one of these options: #{VALID_ATTRIBUTE_NAMES.inspect}" unless options[:params].is_a?(Hash)
343
+ raise "valid params options are: #{VALID_ATTRIBUTE_NAMES.inspect} ... you have invalid params option(s): #{(VALID_ATTRIBUTE_NAMES && options[:params].keys) - VALID_ATTRIBUTE_NAMES}" unless ((VALID_ATTRIBUTE_NAMES && options[:params].keys) - VALID_ATTRIBUTE_NAMES).empty?
344
+ end
345
+
346
+ def self.validate_presence_of_api_key(options) # :nodoc:
347
+ raise "You must define api key. Pillboxr.api_key = 'YOUR SECRET KEY'" unless (self.api_key or options[:params][:key])
348
+ end
349
+
350
+ def self.interpret_params(options = {}) # :nodoc:
351
+ # puts options
352
+ @params = HashWithIndifferentAccess.new(options['params']) || {}
353
+ @params['key'] ||= self.api_key
354
+
355
+ # flex api is crude... this makes it compatible with rails active_resource and will_paginate
356
+ if @params[:start]
357
+ @params['lower_limit'] = (@params[:page] || "0").to_i * @params[:start].to_i
358
+ end
359
+ @params.delete(:page)
360
+ @params.delete(:start)
361
+
362
+ %w(color shape prodcode dea ingredient).each do |param|
363
+ self.send("parse_#{param}".to_sym) if @params[param]
364
+ end
365
+
366
+ @params.delete_if {|k,v| v.nil? }
367
+ options['params'].merge!(@params)
368
+ puts options
369
+ return options
370
+ end
371
+
372
+ def self.parse_color # :nodoc:
373
+ begin
374
+ @params['color'] = case @params['color']
375
+ when NilClass;
376
+ when Array; @params['color'].join(";")
377
+ when /^(\d|[a-f]|[A-F])+/; @params['color'] # valid hex
378
+ else; COLORS[@params['color'].upcase]
379
+ end
380
+ rescue
381
+ # "color not found"
382
+ end
383
+ end
384
+
385
+ def self.parse_shape # :nodoc:
386
+ begin
387
+ @params['shape'] = case @params['shape']
388
+ when NilClass;
389
+ when Array; @params['shape'].join(";")
390
+ when /^([Cc]{1}\d{5})+/; @params['shape'] # valid hex
391
+ else
392
+ SHAPES[@params['shape'].upcase]
393
+ end
394
+ rescue # NoMethodError => e
395
+ # raise X if e.match "shape not found"
396
+ end
397
+ end
398
+
399
+ def self.parse_prodcode # :nodoc:
400
+ # todo: prodcode
401
+ begin
402
+ @params['prodcode'] = case @params['prodcode']
403
+ # when nil; puts "i can see my house! "#raise "Product code cannot be nil" # Schema says PRODUCT_CODE cannot be NULL
404
+ # when Array; params['prodcode'].join(";")
405
+ when /\A(\d{3,}-\d{3,4})\z/; @params['prodcode']
406
+ else;
407
+ end
408
+ rescue
409
+ end
410
+ end
411
+
412
+ def self.parse_dea # :nodoc:
413
+ begin
414
+ @params['dea'] = case @params['dea']
415
+ when NilClass;
416
+ when /^([Cc]{1}\d{5})+/; @params['dea'] # valid hex
417
+ when /\AI{1,3}\z|\AIV\z|\AV\z/; DEA_CODES[@params['dea']]
418
+ else
419
+ raise "Invalid schedule code. Must be one of [I, II, III, IV, V]."
420
+ end
421
+ rescue
422
+ # raise "DEA schedule not found"
423
+ end
424
+ end
425
+
426
+ def self.parse_ingredient # :nodoc:
427
+ begin
428
+ @params['ingredient'] = case @params['ingredient']
429
+ when NilClass;
430
+ when Array; @params['ingredient'].sort.join("; ") # Need to sort alphabetically before send and need a space after semicolon
431
+ when /AND/
432
+ raise "not implemented"
433
+ # Add some parsing to concatenate terms appropriately
434
+ when /OR/
435
+ raise "not implemented"
436
+ # Add some parsing to create two queries, run them, and then merge the results and return it
437
+ else
438
+ # raise "Ingredients not found."
439
+ end
440
+ rescue
441
+ # raise "Ingredient has an invalid format."
442
+ end
443
+ end
444
+ end
445
+
446
+ =begin
447
+ ONE-LINERS
448
+ Pillboxr.api_key = CHANGE_ME_TO_A_VALID_KEY
449
+
450
+ resources = Pill.all(:conditions=>"image_ref is NULL").map(&:name).map{|name| begin Pillboxr.find(:first, :params=>{:has_image=>'1', 'ingredient'=>name.downcase}) rescue name end}; true
451
+ resources = Pill.all.map(&:name).map{|name| begin Pillboxr.find(:first, :params=>{:has_image=>'1', 'ingredient'=>name.downcase}) rescue name end}; true
452
+
453
+ resources.map(&:to_s)
454
+ #=> ["#<Pillboxr:0x2114ea8>", "Imodium", "Penicillin", "Placebo", "Tamiflu", "#<Pillboxr:0x2073044>", "#<Pillboxr:0x2045b08>", "#<Pillboxr:0x2566664>", "Z-Pak", "#<Pillboxr:0x2503fdc>", "#<Pillboxr:0x2489124>", "#<Pillboxr:0x242d770>", "Ex-Lax", "Orlisat"]
455
+
456
+ to_be_filled_out = resources.dup
457
+ to_be_filled_out = to_be_filled_out.reject{|r| r.is_a?(String) }
458
+ to_be_filled_out = to_be_filled_out.reject{|r| r.nil? }
459
+ #to_be_filled_out = to_be_filled_out.reject{|r| r.image_ref.is_a?(String) } # already?
460
+
461
+ #execute
462
+ # all pills
463
+ Pill.all.each {|pill|
464
+ pr = begin
465
+ Pillboxr.find(:first, :params=>{:has_image=>'1', 'prodcode'=>pill.prodcode})
466
+ rescue
467
+ end
468
+ pr ||= begin
469
+ Pillboxr.find(:first, :params=>{:has_image=>'1', 'ingredient'=>pill.name.downcase})
470
+ rescue
471
+ next
472
+ end
473
+ pill.update_attributes :image_ref => pr.image_url('small'),
474
+ :api_ref => pr.api_url,
475
+ :accepted_names => [pr.ingredient, pr.trade_name]
476
+ }
477
+ # Pill.all(:conditions=>'accepted_names is NULL').map(&:name)
478
+
479
+
480
+ resources = Pill.all.map{|pill|
481
+ begin
482
+ r = Pillboxr.find(:first, :params=>{:has_image=>'1', 'ingredient'=>pill.name.downcase})
483
+ pill.update_attributes(
484
+ :image_ref => r.image_url,
485
+ :api_ref => r.api_url,
486
+ :accepted_names => [r.ingredient, r.trade_name]
487
+ )
488
+ rescue
489
+ pill.name
490
+ end
491
+ }; true
492
+
493
+
494
+ #from names
495
+ to_be_filled_out.map {|r|
496
+ p = Pill.find_by_name(r.trade_name);
497
+ if p.nil?
498
+ puts "could not find #{r.trade_name}"
499
+ else
500
+ begin
501
+ p.update_attributes(
502
+ :image_ref => r.image_url,
503
+ :api_ref => r.api_url,
504
+ :accepted_names => [r.ingredient, r.trade_name]
505
+ )
506
+ rescue => e
507
+ puts "pill #{r.trade_name} not updated because: #{e}"
508
+ end
509
+
510
+ end
511
+ }; true
512
+
513
+ # parse out of a messy file
514
+ counter = 1
515
+ names = {}
516
+ Dir.glob('drugnames/*.txt').map {|path| f = File.open(path, "r") {|file|
517
+ while (line = file.gets)
518
+ counter = counter + 1
519
+ names[path] ||= []
520
+ names[path] << line.gsub(","," ").split(" ").first
521
+ end
522
+ puts "Searched #{counter} lines for pill names at the beginning of the line"
523
+ }}
524
+ names
525
+ pill_resources = []
526
+ found_names = []
527
+ names.each {|k,group_names|
528
+ for name in group_names.uniq
529
+ begin
530
+ result = Pillboxr.find(:first, :params=>{:has_image=>'1', 'ingredient'=>name.downcase})
531
+ if result.nil?
532
+ puts "no images found for #{name}"
533
+ else
534
+ pill_resources << result
535
+ found_names << name
536
+ end
537
+ rescue
538
+ puts "could not find #{name}"
539
+ end
540
+ end
541
+ }
542
+ puts "DID find images of the following, stored in 'pill_resources' variable" unless pill_resources.empty?
543
+ for name in found_names
544
+ puts name
545
+ end
546
+
547
+ pill_category = nil #PillCategory.find_by_title("Sexual Health")
548
+ for pill_name in ["","",""]
549
+ Pill.create(:name=>pill_name, :pill_category=>pill_category)
550
+ end
551
+
552
+ =end