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 +10 -3
- data/Gemfile.lock +52 -0
- data/README.md +116 -0
- data/Rakefile +29 -6
- data/lib/pillboxr.rb +35 -532
- data/lib/pillboxr/attributes.rb +267 -0
- data/lib/pillboxr/constants.rb +61 -0
- data/lib/pillboxr/extensions.rb +11 -0
- data/lib/pillboxr/pages.rb +120 -0
- data/lib/pillboxr/params.rb +36 -0
- data/lib/pillboxr/pill.rb +74 -0
- data/lib/pillboxr/request.rb +59 -0
- data/lib/pillboxr/response.rb +8 -0
- data/lib/pillboxr/result.rb +79 -0
- data/lib/pillboxr/version.rb +4 -0
- data/pillboxr.gemspec +18 -7
- data/test/pillboxr/attributes_test.rb +15 -0
- data/test/pillboxr/params_test.rb +28 -0
- data/test/pillboxr/pill_test.rb +20 -0
- data/test/pillboxr/pillboxr_test.rb +167 -0
- data/test/pillboxr/test_helper.rb +10 -0
- metadata +116 -90
- data/.gitignore +0 -6
- data/README.rdoc +0 -46
- data/lib/activeresource_patch.rb +0 -27
- data/lib/compatibility.rb +0 -15
- data/lib/pillbox_resource_noko.rb +0 -40
- data/lib/version.rb +0 -1
- data/test/fixtures/all_meds.yml +0 -8445
- data/test/fixtures/image_meds.yml +0 -4223
- data/test/fixtures/meds.xml +0 -73
- data/test/fixtures/no_image_meds.yml +0 -4223
- data/test/pillboxr_test.rb +0 -106
- data/test/test_helper.rb +0 -16
data/Gemfile
CHANGED
@@ -1,4 +1,11 @@
|
|
1
|
-
source
|
1
|
+
source 'http://rubygems.org'
|
2
2
|
|
3
|
-
|
4
|
-
|
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
|
data/Gemfile.lock
ADDED
@@ -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
|
data/README.md
ADDED
@@ -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
|
-
|
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
|
-
|
9
|
-
|
10
|
-
test.
|
11
|
-
test.
|
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 :
|
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
|
data/lib/pillboxr.rb
CHANGED
@@ -1,552 +1,55 @@
|
|
1
1
|
# -*- encoding: utf-8 -*-
|
2
|
-
$:.unshift(File.dirname(__FILE__))
|
3
2
|
|
4
|
-
|
5
|
-
|
6
|
-
|
7
|
-
|
8
|
-
|
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
|
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
|
-
|
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
|
-
|
61
|
-
|
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
|
-
|
111
|
-
|
112
|
-
|
113
|
-
|
114
|
-
|
115
|
-
|
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
|
-
|
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
|
341
|
-
|
342
|
-
|
343
|
-
|
344
|
-
|
345
|
-
|
346
|
-
|
347
|
-
|
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
|
-
|
481
|
-
|
482
|
-
|
483
|
-
|
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
|
-
|
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
|