pillboxr 0.0.3 → 0.6.0

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/Gemfile CHANGED
@@ -1,4 +1,11 @@
1
- source :gemcutter
1
+ source 'http://rubygems.org'
2
2
 
3
- # Specify your gem's dependencies in pillbox.gemspec
4
- gemspec
3
+ gem 'activeresource', '~> 3.2.0'
4
+ gem 'httparty'
5
+
6
+ group :development do
7
+ gem 'guard'
8
+ gem 'guard-minitest'
9
+ gem 'vcr'
10
+ gem 'webmock'
11
+ end
@@ -0,0 +1,52 @@
1
+ GEM
2
+ remote: http://rubygems.org/
3
+ specs:
4
+ activemodel (3.2.6)
5
+ activesupport (= 3.2.6)
6
+ builder (~> 3.0.0)
7
+ activeresource (3.2.6)
8
+ activemodel (= 3.2.6)
9
+ activesupport (= 3.2.6)
10
+ activesupport (3.2.6)
11
+ i18n (~> 0.6)
12
+ multi_json (~> 1.0)
13
+ addressable (2.2.8)
14
+ builder (3.0.0)
15
+ crack (0.3.1)
16
+ ffi (1.1.2)
17
+ guard (1.2.3)
18
+ listen (>= 0.4.2)
19
+ thor (>= 0.14.6)
20
+ guard-minitest (0.5.0)
21
+ guard (>= 0.4)
22
+ httparty (0.8.3)
23
+ multi_json (~> 1.0)
24
+ multi_xml
25
+ i18n (0.6.0)
26
+ listen (0.4.7)
27
+ rb-fchange (~> 0.0.5)
28
+ rb-fsevent (~> 0.9.1)
29
+ rb-inotify (~> 0.8.8)
30
+ multi_json (1.3.6)
31
+ multi_xml (0.5.1)
32
+ rb-fchange (0.0.5)
33
+ ffi
34
+ rb-fsevent (0.9.1)
35
+ rb-inotify (0.8.8)
36
+ ffi (>= 0.5.0)
37
+ thor (0.15.4)
38
+ vcr (2.2.4)
39
+ webmock (1.8.8)
40
+ addressable (~> 2.2.8)
41
+ crack (>= 0.1.7)
42
+
43
+ PLATFORMS
44
+ ruby
45
+
46
+ DEPENDENCIES
47
+ activeresource (~> 3.2.0)
48
+ guard
49
+ guard-minitest
50
+ httparty
51
+ vcr
52
+ webmock
@@ -0,0 +1,116 @@
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](http://pillbox.nlm.nih.gov).
4
+
5
+ The pillbox API provides information from the FDA about various prescription medications.
6
+
7
+ The current version of this library has two forms. The first (preferred) version does not depend on any gems except for `httparty`. The second version depends upon `active_resource`. This version of Pillboxr inherits from ActiveResource to perform its XML wrapping so ActiveResource 3.2.6 is a requirement for using the wrapper. This version will is deprecated and will be removed in the future. It is not included in the installed gem version, but the code is available in this repository mostly for historical interest.
8
+
9
+ *Note:* This library is designed for use with Ruby 1.9.3 and above, and will not work with earlier versions of Ruby.
10
+
11
+ ***
12
+
13
+ ## Usage
14
+
15
+ Getting started is fairly easy:
16
+
17
+ $ gem install pillboxr
18
+
19
+ Next obtain an API key and paste it into a file called `api_key.yml` in the root directory of your project. See below for directions on obtaining an API key.
20
+
21
+ Finally:
22
+
23
+ ```ruby
24
+ require 'pillboxr' # You may have to require rubygems first
25
+
26
+ result = Pillboxr.with({:color => :blue, :image => true}) # Get result object with one page of blue pills with images.
27
+
28
+ result.pages.current.pills # An array with the retrieved pill objects.
29
+ ```
30
+
31
+ ###### or
32
+
33
+ ```ruby
34
+ require 'pillboxr'
35
+
36
+ result = Pillboxr.color(:blue).image(true).all # Get result of object with one page of blue pills with images associated.
37
+
38
+ result.pages.current.pills # an array with the retrieved pill objects.
39
+ ```
40
+
41
+ ***
42
+
43
+ **Important:** *When chaining query methods you must add the `all` method on the end of the query chain, similar to working with `ActiveRelation` in Rails, so the request can be lazily evaluated.*
44
+
45
+ ***
46
+
47
+ Both query methods also have block forms that allow fetching of additional result pages. For example:
48
+
49
+ ```ruby
50
+ require 'pillboxr'
51
+
52
+ result = Pillboxr.with({:color => :blue}) do |r|
53
+ r.pages.each do |page|
54
+ page.get unless page.retrieved?
55
+ end
56
+ end
57
+
58
+ all_blue_pills = []
59
+ result.pages.each { |page| all_blue_pills << page.pills }
60
+
61
+ all_blue_pills.flatten! # all_blue_pills is now an array of all 2059 blue pills.
62
+ ```
63
+
64
+ ###### or
65
+
66
+ ```ruby
67
+ require 'pillboxr'
68
+
69
+ result = Pillboxr.color(:blue).all do |r|
70
+ r.pages.each do |page|
71
+ page.get unless page.retrieved?
72
+ end
73
+ end
74
+
75
+ all_blue_pills = []
76
+ result.pages.each { |page| all_blue_pills << page.pills }
77
+
78
+ all_blue_pills.flatten! # all_blue_pills is now an array of all 2059 blue pills.
79
+ ```
80
+
81
+ You can run the tests by typing `rake` in the library directory. You may have to install some development gems prior to running the tests by running `bundle install` in the library directory.
82
+
83
+ ***
84
+
85
+ The hash passed to the `with` method may include any of the following parameters:
86
+
87
+ ```ruby
88
+ :color => Symbol or Array with multiple colors (see http://pillbox.nlm.nih.gov/API-documentation.html)
89
+ :score => Boolean
90
+ :ingredient => Symbol or Array with multiple ingredients (returned results include all ingredients)
91
+ :inactive => Symbol
92
+ :dea => Symbol or any of 'I, II, III, IV, V'
93
+ :author => String
94
+ :shape => Symbol (Shape or Hex)
95
+ :imprint => Symbol
96
+ :prodcode => Symbol (Product Code: see http://pillbox.nlm.nih.gov/API-documentation.html)
97
+ :image => Boolean
98
+ :size => Integer for size in millimeters (currently this encompasses a range of +/- 2 mm)
99
+ :lower_limit => Integer for which returned record to start at
100
+ ```
101
+
102
+ Please see specific files or the document directory for specific usage examples. Further API documentation available on the [project homepage](http://pillbox.nlm.nih.gov/NLM_Pillbox_API_documentation_v2_2011.09.27.pdf) (PDF link)
103
+
104
+ ## KNOWN BUGS
105
+
106
+ * The library allows you to request the same page repeatedly resulting in duplicate data.
107
+
108
+ * Please note that some XML in the Pillbox API is unescaped.
109
+
110
+ * Please report additional bugs via [github issues](https://github.com/kgautreaux/pillboxr/issues).
111
+
112
+ API provided through the generous support by the FDA in both money and resources. Work conducted by NLM at NIH.
113
+
114
+ Please contact david.hale at nlm.nih.gov for an api key. There is no bandwidth limit currently.
115
+
116
+ Data is owned by companies, mandatorily licenced for X purposes.
data/Rakefile CHANGED
@@ -1,15 +1,38 @@
1
- require 'rubygems'
1
+ # -*- encoding: utf-8 -*-
2
2
  require 'rake'
3
3
  require 'rake/testtask'
4
4
  require 'bundler'
5
+ begin
6
+ require 'pry'
7
+ rescue LoadError
8
+ require 'irb'
9
+ end
5
10
 
6
11
  Bundler::GemHelper.install_tasks
7
12
 
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'
13
+ Rake::TestTask.new(:standalone_test) do |test|
14
+ test.libs << 'lib' << 'test/pillboxr'
15
+ test.pattern = 'test/pillboxr/**/*_test.rb'
16
+ test.verbose = true
17
+ end
18
+
19
+ Rake::TestTask.new(:activeresource_test) do |test|
20
+ test.libs = []
21
+ test.libs << 'lib/active_resource' << 'test/pillboxr_activeresource'
22
+ test.pattern = 'test/pillboxr_activeresource/**/*_test.rb'
12
23
  test.verbose = true
13
24
  end
14
25
 
15
- task :default => :test
26
+ task :console do
27
+ if Kernel.const_defined?(:Pry)
28
+ Pry.start
29
+ else
30
+ IRB.start
31
+ end
32
+ end
33
+
34
+ task :c => :console
35
+
36
+ task :all => [:standalone_test, :activeresource_test]
37
+
38
+ task :default => :standalone_test
@@ -1,552 +1,55 @@
1
1
  # -*- encoding: utf-8 -*-
2
- $:.unshift(File.dirname(__FILE__))
3
2
 
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'
3
+ require_relative 'pillboxr/extensions'
4
+ require_relative 'pillboxr/result'
5
+ require_relative 'pillboxr/pill'
6
+ require_relative 'pillboxr/params'
7
+ require_relative 'pillboxr/request'
30
8
 
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
9
+ module Pillboxr
39
10
 
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
11
+ def complete(params = @params, &block)
12
+ return Result.new(Request.new(params).perform, &block)
49
13
  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
14
 
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
15
+ def with(query_hash, &block)
16
+ @params ||= Params.new(self)
109
17
 
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."
18
+ query_hash.each do |k,v|
19
+ if attributes.keys.include?(k)
20
+ @params << symbol_to_instance(k,v)
21
+ elsif api_attributes.keys.include?(k)
22
+ puts "#{api_attributes.fetch(k)} => #{v}"
23
+ @params << symbol_to_instance(api_attributes.fetch(k),v)
212
24
  else
213
- raise
25
+ raise "Invalid attributes hash."
26
+ next
214
27
  end
215
28
  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
29
 
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
30
+ complete(@params, &block)
242
31
  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
32
 
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
33
+ def respond_to_missing?(method_name, include_private = false) # :nodoc:
34
+ (attributes.keys.include?(method_name) || attributes.values.include?(method_name))
307
35
  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
36
 
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."
37
+ def method_missing(method_name, *args, &block) # :nodoc:
38
+ @params ||= Params.new(self)
39
+ if attributes.keys.include?(method_name)
40
+ @params << symbol_to_instance(method_name, args.first)
41
+ elsif api_attributes.keys.include?(method_name)
42
+ @params << symbol_to_instance(api_attributes.fetch(method_name), args.first)
43
+ else
44
+ super
442
45
  end
443
46
  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
47
 
48
+ private
479
49
 
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
-
50
+ def symbol_to_instance(symbol, value)
51
+ klass = String(symbol).gsub(/_/, "").capitalize
52
+ klass.extend(Pillboxr::Extensions) unless klass.methods.include?(:to_constant)
53
+ klass.to_constant.new(value)
510
54
  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
55
+ end