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 +4 -0
- data/Gemfile +4 -0
- data/Gemfile.lock +27 -0
- data/README.rdoc +44 -0
- data/Rakefile +15 -0
- data/lib/activeresource_patch.rb +27 -0
- data/lib/compatibility.rb +15 -0
- data/lib/pillbox_resource_noko.rb +40 -0
- data/lib/pillboxr.rb +552 -0
- data/lib/version.rb +1 -0
- data/pillboxr.gemspec +28 -0
- data/test/fixtures/all_meds.yml +8445 -0
- data/test/fixtures/image_meds.yml +4223 -0
- data/test/fixtures/meds.xml +73 -0
- data/test/fixtures/no_image_meds.yml +4223 -0
- data/test/pillboxr_test.rb +106 -0
- data/test/test_helper.rb +16 -0
- metadata +131 -0
data/.gitignore
ADDED
data/Gemfile
ADDED
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,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
|