crowdflower 0.11.0 → 0.12.0

Sign up to get free protection for your applications and to get access to all the features.
@@ -0,0 +1,7 @@
1
+ ---
2
+ SHA1:
3
+ metadata.gz: 76063401ad10a5f00f6eef25ce35321d57b0ac62
4
+ data.tar.gz: 4e9b04672d27280fdfde741f63ec4217731dd7f0
5
+ SHA512:
6
+ metadata.gz: 82241d2fd4dfe9df428181d8ca11edb0d830dcdfe253c4eff9b2e989bfdd83f4379af59f414b08a02316823e0ada074c5867d5f34c6b80418a6348e53dc67df2
7
+ data.tar.gz: 93914eb90b7c1fa4f044e536e7bd027fef710bc6c9378801ab84bb16e92e9299b0326b9a2959d7f5d8da725f6306069a4f0c6fb143718e61407d197444a18cd4
data/README.md CHANGED
@@ -40,7 +40,7 @@ CrowdFlower.connect!( 'CrowdFlower.yaml' )
40
40
  ```
41
41
 
42
42
  ## Usage and Examples
43
- #####[Example Job](https://api.crowdflower.com/v1/jobs/418404/) - referenced throught the following examples (must be signed in to view).
43
+ #####This [job](https://api.crowdflower.com/v1/jobs/418404/) is referenced throughout the following examples (must be signed in to view).
44
44
 
45
45
  ### Access Job Info
46
46
 
@@ -76,17 +76,17 @@ require 'crowdflower'
76
76
 
77
77
  API_KEY = "YOUR_API_KEY"
78
78
  DOMAIN_BASE = "https://api.crowdflower.com"
79
- JOB_ID = 418404
80
79
 
81
80
  CrowdFlower::Job.connect! API_KEY, DOMAIN_BASE
82
81
 
82
+ job_id = 418404
83
83
  job_one = CrowdFlower::Job.new(job_id)
84
84
  job_two = job_one.copy
85
85
  ```
86
86
 
87
87
  ### Available Features (Methods)
88
88
 
89
- #####GET - https://crowdflower.com/jobs/418404.json
89
+ #####GET: Example job's JSON, which the GET method has access to: https://crowdflower.com/jobs/418404.json
90
90
 
91
91
  ```ruby
92
92
  job.get["css"]
@@ -141,22 +141,33 @@ job.get["custom_key"]
141
141
  job.get["options"]
142
142
  ```
143
143
 
144
- #####UPLOAD (data to create units)
144
+ #####UPLOAD: upload a file to create the data (or units) in your job.
145
145
 
146
146
  ```ruby
147
+ # opts can be used to force an upload
147
148
  job.upload(filename, type, opts)
148
149
  job.upload("crowdshopping.csv", "text/csv")
149
150
  ```
150
151
 
151
- #####CHANNELS - http://api.crowdflower.com/v1/jobs/418404/channels
152
+ #####CHANNELS: http://api.crowdflower.com/v1/jobs/418404/channels
152
153
 
153
154
  ```ruby
154
- job.channels
155
+ # view all enabled channels and available channels:
156
+ job.channels
157
+
158
+ # view only the enabled ones:
159
+ job.channels["enabled_channels"]
160
+
161
+ # turn on specific channels:
155
162
  job.enable_channels(channels)
156
- job.enable_channels("cf_internal")
163
+ job.enable_channels(["4x4bux_com", "abitback", "aceinnovations"])
164
+
165
+ # turn off a specific channel (does not take an array like enable_chanels):
166
+ job.disable_channel(channel_name)
167
+ job.disable_channel("4x4bux_com")
157
168
  ```
158
169
 
159
- #####TAGS - https://api.crowdflower.com/jobs/418404/tags
170
+ #####TAGS: https://api.crowdflower.com/jobs/418404/tags
160
171
 
161
172
  ```ruby
162
173
  tags = "shoes", "shopping", "fashion"
@@ -199,8 +210,13 @@ unit.judgments(444154130)
199
210
  #####UNIT.CREATE: Create a new unit.
200
211
 
201
212
  ```ruby
213
+ unit.create(data, gold = false)
214
+
215
+ # normal unit
202
216
  unit.create("glitter_color"=>"blue")
203
- unit.create("glitter_color"=>"blue", gold: true)
217
+
218
+ # test question (gold) unit
219
+ unit.create("glitter_color"=>"blue", true)
204
220
  ```
205
221
 
206
222
  #####UNIT.COPY: Copy an existing unit.
@@ -213,7 +229,7 @@ unit.copy(444154130, 418404, "glitter_color"=>"blue")
213
229
  #####UNIT.SPLIT: In cases where multiple values are stored in the cells of the same column, you can use the Split Column function to parse the data into two or more columns by specifying a delimiter (most typically a newline character).
214
230
 
215
231
  ```ruby
216
- unit.split(on, with = " ")
232
+ unit.split(on, with = ", ")
217
233
  ```
218
234
 
219
235
  #####UNIT.UPDATE: Update the value of a unit's key.
@@ -241,7 +257,7 @@ unit.cancel(unit_id)
241
257
  unit.delete(unit_id)
242
258
  ```
243
259
 
244
- #####UNIT.REQUEST_MORE_JUDGMENTS: Get more answers (judgments) for a specific unit by using this method with the unit's id and the amount of requested judgments passed as params.
260
+ #####UNIT.REQUEST_MORE_JUDGMENTS: Pass in the unit id and the number of additional judgments needed.
245
261
 
246
262
  ```ruby
247
263
  unit.request_more_judgments(unit_id, nb_judgments = 1)
@@ -253,29 +269,26 @@ unit.request_more_judgments(unit_id, nb_judgments = 1)
253
269
  order = CrowdFlower::Order.new(job)
254
270
  ```
255
271
 
256
- #####ORDER.DEBIT: When a job has data (units) and properly working cml, it is ready to launch. The launch tab on the job dashboard is the same as calling order.debit.
272
+ #####ORDER.DEBIT: This is the same as clicking launch from the job dashboard; your job must have at least 5 units and CML form elements for this to work.
257
273
 
258
274
  ```ruby
259
- # defaults to on_demand channels
260
- order.debit(units_count, channels = ["on_demand"])
261
-
262
- # pass in specific channel name to launch job w/ that channel
275
+ order.debit(units_count, channels)
263
276
  order.debit(6, "cf_internal")
264
277
  ```
265
278
 
266
- #####PAUSE: Only can be called on running jobs.
279
+ #####PAUSE: Can only be called on running jobs.
267
280
 
268
281
  ```ruby
269
282
  job.pause
270
283
  ```
271
284
 
272
- #####RESUME: Only can be called on paused or completed jobs.
285
+ #####RESUME: Can only be called on paused or completed jobs.
273
286
 
274
287
  ```ruby
275
288
  job.resume
276
289
  ```
277
290
 
278
- #####CANCEL: Only can be called on paused jobs.
291
+ #####CANCEL: Can only be called on paused jobs.
279
292
 
280
293
  ```ruby
281
294
  job.cancel
@@ -300,60 +313,64 @@ job.delete
300
313
  worker = CrowdFlower::Worker.new(job)
301
314
  ```
302
315
 
303
- #####WORKER.BONUS: Award a bonus in cents, 200 for $2.00 and (optionally) add a message
316
+ #####WORKER.BONUS: Award a bonus in cents, 200 for $2.00 and optionally, add a message.
304
317
 
305
318
  ```ruby
306
319
  worker.bonus(worker_id, amount, reason=nil)
307
- worker.bonus(23542619, 200, "You shoe shop like a pro! Thanks for the awesome answers!")
320
+ worker.bonus(99999999, 200, "You shoe shop like a pro! Here's a bonus for the awesome answers!")
308
321
  ```
309
322
 
310
- #####WORKER.REJECT: Stops contributors from completing tasks, and removes the contributors judgments from a job. Try to only use when a job is running, otherwise the completed job will lose judgments and be unable to collect replacement ones.
323
+ #####WORKER.REJECT: This method is only available to Pro and Enterprise users. Calling worker.reject stops a contributor from completing tasks and removes the contributor's judgments. It is best used when a job is still running as a completed job cannot collect new judgments to replace the rejected ones.
311
324
 
312
325
  ```ruby
313
326
  worker.reject(worker_id)
314
- worker.reject(14952322)
327
+ worker.reject(99999999)
315
328
  ```
316
329
 
317
- #####WORKER.NOTIFY: Sends a notification to a specific contributor. The contributor will see the message under their notifications.
330
+ #####WORKER.NOTIFY: Sends a message to the specified contributor; appears in the contributor's dashboard notifications.
318
331
 
319
332
  ```ruby
320
- worker.notify(worker_id, subject, message)
321
- worker.notify(23542619, "you earned a bonus!", "good job!")
333
+ worker.notify(worker_id, message)
334
+ worker.notify(99999999, "Thanks for working on this task!")
322
335
  ```
323
336
 
324
- #####WORKER.FLAG: Stops contributors from completing tasks. Their judgments remain in their current state of tainted or non-tainted and will not be thrown away.
337
+ #####WORKER.FLAG: Prevents a contributor from completing a task; judgments remain in their current state and will not be thrown away.
325
338
 
326
339
  ```ruby
327
- worker.flag(worker_id, reason=nil)
328
- worker.flag(14952322, "testing")
340
+ worker.flag(worker_id, reason = nil, persist = false)
341
+ worker.flag(99999999, "Flagging worker from this job.")
342
+
343
+ # persist = true flags the contributor from all of your jobs
344
+ worker.flag(worker_id, reason = nil, persist = true)
345
+ worker.flag(99999999, "Flaging worker from all my jobs.")
329
346
  ```
330
347
 
331
- #####WORKER.DEFLAG: Allows a flagged contributor to continue completing tasks.
348
+ #####WORKER.DEFLAG: Removes flag and allows contributor to continue work.
332
349
  ```ruby
333
- worker.deflag(worker_id)
334
- worker.deflag(14952322)
350
+ worker.deflag(worker_id, reason)
351
+ worker.deflag(99999999, "Worker was mistakenly flagged.")
335
352
  ```
336
353
 
337
- #####JUDGMENTS - http://api.crowdflower.com/v1/jobs/418404/units/judgments
354
+ #####JUDGMENTS: http://api.crowdflower.com/v1/jobs/418404/units/judgments
338
355
 
339
356
  ```ruby
340
357
  judgment = CrowdFlower::Judgment.new(job)
341
358
  judgment.all
342
359
  judgment.get(judgment_id)
343
- judgment.get(1239592918)
360
+ judgment.get(9999999999)
344
361
 
345
362
  # Return every judgment for the given unit
346
363
  job.units.judgments(unit_id_number)
347
- job.units.judgments(444154130)
364
+ job.units.judgments(9999999999)
348
365
  ```
349
366
 
350
- #####LEGEND: Returns all of the cml in a job; shows all questions and available answers, see example: http://api.crowdflower.com/v1/jobs/418404/legend.
367
+ #####LEGEND: Returns all the job's CML - http://api.crowdflower.com/v1/jobs/418404/legend
351
368
 
352
369
  ```ruby
353
370
  job.legend
354
371
  ```
355
372
 
356
- #####STATUS: parsed JSON response or access attributes like GET
373
+ #####STATUS: Returns a list of JSON unit and judgment attributes.
357
374
 
358
375
  ```ruby
359
376
  job.status
@@ -368,11 +385,16 @@ job.status["completed_gold_estimate"]
368
385
  job.status["ordered_units"]
369
386
  ```
370
387
 
371
- #####DOWNLOAD_CSV: Downloads a CSV of the job with results, sometimes as csv and sometimes as a zip containing the CSV.
388
+ #####DOWNLOAD_CSV: Download a zip file containing a CSV or JSON report, depending which one you specify. Below are examples of the available reports:
372
389
 
373
390
  ```ruby
374
- job.download_csv(type, filename, opts)
375
- job.download_csv(full, nil, force:true)
391
+ job.download_csv(type, filename)
392
+ job.download_csv(:full, 'full_report.zip')
393
+ job.download_csv(:aggregated, 'aggregated_report.zip')
394
+ job.download_csv(:source, 'source_report.zip')
395
+ job.download_csv(:gold_report, 'gold_report.zip')
396
+ job.download_csv(:workset, 'workset_report.zip')
397
+ job.download_csv(:json, 'json_report.zip')
376
398
  ```
377
399
 
378
400
  ## Helpful Documentation Links
data/VERSION CHANGED
@@ -1 +1 @@
1
- 0.11.0
1
+ 0.12.0
@@ -0,0 +1,5 @@
1
+ # This file can be used for running your job and for running the integration tests
2
+ # Rename this yaml template to "account_info.yml" or update the .gitignore file so it will not be pushed to github
3
+
4
+ API_KEY: add your api key here
5
+ PROJECT_NUM: add your project number here
@@ -2,14 +2,16 @@
2
2
  # DO NOT EDIT THIS FILE DIRECTLY
3
3
  # Instead, edit Jeweler::Tasks in Rakefile, and run 'rake gemspec'
4
4
  # -*- encoding: utf-8 -*-
5
+ # stub: crowdflower 0.12.0 ruby lib
5
6
 
6
7
  Gem::Specification.new do |s|
7
8
  s.name = "crowdflower"
8
- s.version = "0.11.0"
9
+ s.version = "0.12.0"
9
10
 
10
11
  s.required_rubygems_version = Gem::Requirement.new(">= 0") if s.respond_to? :required_rubygems_version=
12
+ s.require_paths = ["lib"]
11
13
  s.authors = ["Brian P O'Rourke", "Chris Van Pelt"]
12
- s.date = "2014-05-02"
14
+ s.date = "2014-05-05"
13
15
  s.description = "A toolkit for interacting with CrowdFlower via the REST API.\n\nThis is alpha software. Have fun!\n\n"
14
16
  s.email = "brian@crowdflower.com"
15
17
  s.extra_rdoc_files = [
@@ -26,6 +28,7 @@ Gem::Specification.new do |s|
26
28
  "VERSION",
27
29
  "bindev/cl_skel.rb",
28
30
  "bindev/crowdflower.rb",
31
+ "config/account_info.sample.yml",
29
32
  "crowdflower.gemspec",
30
33
  "lib/crowdflower.rb",
31
34
  "lib/crowdflower/base.rb",
@@ -34,16 +37,15 @@ Gem::Specification.new do |s|
34
37
  "lib/crowdflower/order.rb",
35
38
  "lib/crowdflower/unit.rb",
36
39
  "lib/crowdflower/worker.rb",
37
- "test/integration_tests.rb",
38
- "test/sample.csv"
40
+ "test/crowdshopping.csv",
41
+ "test/integration_tests.rb"
39
42
  ]
40
43
  s.homepage = "http://github.com/dolores/ruby-crowdflower"
41
- s.require_paths = ["lib"]
42
- s.rubygems_version = "1.8.23"
44
+ s.rubygems_version = "2.2.0"
43
45
  s.summary = "a toolkit for the CrowdFlower API"
44
46
 
45
47
  if s.respond_to? :specification_version then
46
- s.specification_version = 3
48
+ s.specification_version = 4
47
49
 
48
50
  if Gem::Version.new(Gem::VERSION) >= Gem::Version.new('1.2.0') then
49
51
  s.add_runtime_dependency(%q<httparty>, [">= 0.7.4"])
@@ -127,6 +127,10 @@ module CrowdFlower
127
127
  connection.post("#{resource_uri}/#{@id}/channels", {:body => { :channels => channels } } )
128
128
  end
129
129
 
130
+ def disable_channel(channel_name)
131
+ connection.post("#{resource_uri}/#{@id}/disable_channel", {:body => { :channel_name => channel_name } } )
132
+ end
133
+
130
134
  def tags
131
135
  connection.get("#{resource_uri}/#{@id}/tags")
132
136
  end
@@ -0,0 +1,7 @@
1
+ Brand Name,Shoe Style,Shoe Type,Glitter Color,Image,Retailer Name,Retailer URL
2
+ Jeffery Campbell,Lita,Booties,Red,"",,
3
+ Jeffery Campbell,Lita,Booties,Gold,"",,
4
+ Jeffery Campbell,Lita,Booties,Multi,https://encrypted-tbn1.gstatic.com/images?q=tbn:ANd9GcTocPVxYbL4Jeb6vQy-D3fPMXFHStsjYp43fMWb_ocHsgEMQQfGww,,
5
+ Jeffery Campbell,Lita,Booties,Silver,"",,
6
+ Jeffery Campbell,Lita,Booties,Black,"",,
7
+ Jeffery Campbell,Lita,Booties,Other,"",,
@@ -1,11 +1,12 @@
1
1
  $: << File.dirname(__FILE__) + "/../lib"
2
2
 
3
- require 'rubygems'
4
3
  require 'crowdflower'
5
4
  require 'json'
6
5
 
7
- API_KEY = ENV["API_KEY"]
8
- DOMAIN_BASE = ENV["DOMAIN_BASE"] || "https://api.localdev.crowdflower.com:8443"
6
+ CONFIG = YAML::load(File.read("config/account_info.yml"))
7
+ API_KEY = CONFIG["API_KEY"]
8
+ PROJECT_NUM = CONFIG["PROJECT_NUM"]
9
+ DOMAIN_BASE = "https://api.crowdflower.com" || "https://api.localdev.crowdflower.com:8443"
9
10
 
10
11
  CrowdFlower::Job.connect! API_KEY, DOMAIN_BASE
11
12
  unless API_KEY && API_KEY.size > 3
@@ -23,14 +24,10 @@ EOF
23
24
  exit 1
24
25
  end
25
26
 
26
- # If you turn this on, tasks will be posted on CrowdFlower and your
27
- # account will be charged. This is inadvisable for anyone other than
28
- # CrowdFlower employees.
29
- I_AM_RICH = ENV["CF_LIVE_TRANSACTIONS"] == "true"
27
+ RUNNING_TESTS = ENV["CF_LIVE_TRANSACTIONS"] == "true"
30
28
 
31
- if I_AM_RICH
32
- puts "*** LIVE TRANSACTIONS ENABLED - THIS TEST RUN WILL BE CHARGED ***"
33
- puts
29
+ if RUNNING_TESTS
30
+ puts "*** LIVE TRANSACTIONS ENABLED - THIS TEST RUN WILL BE CHARGED TO YOUR ACCOUNT ***"
34
31
  end
35
32
 
36
33
  def wait_until
@@ -62,111 +59,253 @@ def say(msg)
62
59
  $stdout.puts msg
63
60
  end
64
61
 
62
+ #################################################
63
+ # API CONNECTION
64
+ #################################################
65
+ say "Connecting to the API"
66
+ CrowdFlower.connect! API_KEY, DOMAIN_BASE
65
67
 
66
- say "defining multiple api keys"
67
- (job_subclass_with_valid_custom_key = Class.new(CrowdFlower::Job)).connect! API_KEY, DOMAIN_BASE
68
- (job_subclass_with_invalid_custom_key = Class.new(CrowdFlower::Job)).connect! 'invalid api key', DOMAIN_BASE
69
- job_subclass_with_no_custom_key = Class.new(CrowdFlower::Job)
68
+ #################################################
69
+ # CREATE JOB
70
+ #################################################
71
+ say "Creating a blank job."
72
+ job_1 = CrowdFlower::Job.create("Job_1: Connection Check")
73
+ p "job_1 id: #{job_1.get["id"]}"
74
+ p "job_1 units_count: #{job_1.get["units_count"]}"
75
+
76
+ #################################################
77
+ # CHECK/ ENABLE CHANNELS
78
+ #################################################
79
+ say "Checking that job_1 does not have any enabled channels."
80
+ assert job_1.channels["enabled_channels"].empty?
81
+
82
+ say "Enabling the cf_internal channel."
83
+ job_1.enable_channels("cf_internal")
84
+ assert job_1.channels["enabled_channels"] == ["cf_internal"]
85
+
86
+ #################################################
87
+ # UPLOAD DATA/ CREATE NEW JOB
88
+ #################################################
89
+ say "Uploading CSV to create job_2."
90
+ job_2 = CrowdFlower::Job.upload(File.dirname(__FILE__) + "/crowdshopping.csv", "text/csv")
91
+ job_2_id = job_2.get["id"]
92
+ p "job_2 id: #{job_2_id}"
93
+
94
+ #################################################
95
+ # ADD UNITS
96
+ #################################################
97
+ say "-- Waiting for CrowdFlower to process the data."
98
+ wait_until { job_2.get["units_count"] == 6 }
99
+
100
+ say "Adding some more data."
101
+ job_2.upload(File.dirname(__FILE__) + "/crowdshopping.csv", "text/csv")
102
+
103
+ say "-- Waiting for CrowdFlower to process the data."
104
+ wait_until { job_2.get["units_count"] == 12 }
105
+
106
+ #################################################
107
+ # PING UNITS
108
+ #################################################
109
+ say "Pinging job_2 units."
110
+ assert job_2.units.ping['count'] == 12
111
+ assert job_2.units.ping['done'] == true
112
+
113
+ #################################################
114
+ # COUNT UNITS
115
+ #################################################
116
+ say "Checking for 12 units in job_2."
117
+ assert job_2.units.all.size == 12
118
+
119
+ #################################################
120
+ # UPDATE JOB_2
121
+ #################################################
122
+ say "Adding title, project number, instructions, CML"
123
+ job_2.update({:title => 'Job_2: CrowdShopping',
124
+ :project_number => PROJECT_NUM,
125
+ :instructions => '<p>There are six questions to this task. In this order, the questions ask if you were able to find a pair of Lita shoes for sale in red glitter, gold glitter, multi glitter, silver glitter, black glitter, or other color of glitter.</p><p>There is a photo of the shoe in correlating color as the question right below the question. It will give you a better idea of what to look for.</p>',
126
+ :cml => '<cml:radios label="Were you able to find an online retailer selling Jeffery Campbell Lita Booties in {{glitter_color}}?" validates="required" name="color_found" instructions="If you found the shoes we are looking for, click yes to fill in the website url."><p class="shoe-img">Example Photo: <img src="{{image}}" width="100" /></p><cml:radio label="Yes, I found an online retailer selling Lita shoes in {{glitter_color}}." value="yes"></cml:radio><cml:radio label="No, I could not find an online retailer selling Lita shoes in {{glitter_color}}." value="no"></cml:radio></cml:radios><br /><cml:text label="Please enter the name of the online retailer." default="Example: Karmaloop" validates="required" only-if="color_found:[yes]" name="please_enter_the_name_of_the_online_retailer"></cml:text><br /><cml:text label="Please enter the url to the shoes you found." default="Example: www.karmaloop.com/jeffery-campbell-litas-multiglitter" validates="required url" only-if="color_found:[yes]" name="please_enter_the_url_to_the_shoes_you_found"></cml:text>'})
127
+
128
+ #################################################
129
+ # ADD/ UPDATE/ REMOVE TAGS
130
+ #################################################
131
+ say "Checking if tags exist."
132
+ assert job_2.tags.empty?
133
+
134
+ say "Adding 'shoes' and 'glitter' to tags."
135
+ job_2.add_tags ["shoes", "glitter"]
136
+ assert job_2.tags.map{|t| t["name"]}.sort == ["glitter", "shoes"]
137
+
138
+ say "Removing 'shoes' tag."
139
+ job_2.remove_tags ["shoes"]
140
+ assert job_2.tags.map{|t| t["name"]} == ["glitter"]
141
+
142
+ say "Updating tags to 'fashion' 'fun' and 'glitter'."
143
+ job_2.update_tags ["fashion", "fun", "glitter"]
144
+ assert job_2.tags.map{|t| t["name"]} == ["fashion", "fun", "glitter"]
145
+
146
+ #################################################
147
+ # CHECK JOB_2 CHANNELS
148
+ #################################################
149
+ say "Checking that channels are turned on."
150
+ assert !job_2.channels["enabled_channels"].empty?
151
+ p "job_2 enabled_channels: #{job_2.channels["enabled_channels"]}"
152
+
153
+ #################################################
154
+ # ORDER JOB
155
+ #################################################
156
+ say "Ordering (launching) job_2 with 12 units."
157
+ order = CrowdFlower::Order.new(job_2)
158
+ order.debit(12, ["4x4bux_com", "bitcoinget", "cf_internal"])
159
+ wait_until { job_2.get["state"].casecmp('running') == 0}
160
+ # turning channels from on_demand work force adds "on_demand" to enabled_channels
161
+ assert job_2.channels["enabled_channels"] == ["4x4bux_com", "bitcoinget", "cf_internal", "on_demand"]
162
+
163
+ #################################################
164
+ # DISABLE A CHANNEL
165
+ #################################################
166
+ say "Disabling 'bitcoinget' channel."
167
+ job_2.disable_channel("bitcoinget")
168
+ assert job_2.channels["enabled_channels"] == ["4x4bux_com", "cf_internal", "on_demand"]
169
+
170
+ #################################################
171
+ # UNIT METHODS
172
+ #################################################
173
+ say "Setting up units."
174
+ unit_1 = job_2.units.all.to_a[0][0]
175
+ unit_2 = job_2.units.all.to_a[1][0]
176
+
177
+ unit = CrowdFlower::Unit.new(job_2)
178
+ wait_until { unit.get(unit_1)["state"] == "judgable" }
179
+ p "unit_1 id: #{unit_1}"
180
+ wait_until { unit.get(unit_1)["state"] == "judgable" }
181
+ p "unit_2 id: #{unit_2}"
182
+
183
+ say "Making unit_1 a test question (gold)."
184
+ unit.make_gold(unit_1)
185
+ assert unit.get(unit_1)["state"] == "golden"
186
+
187
+ say "Copying unit_2."
188
+ unit.copy(unit_2, job_2_id, "glitter_color"=>"blue")
189
+ assert job_2.get["units_count"] == 13
190
+
191
+ say "Creating a new test question unit."
192
+ unit.create({"glitter_color"=>"white"}, true)
193
+ assert job_2.get["units_count"] == 14
194
+ assert job_2.get["golds_count"] == 2
195
+
196
+ say "Canceling unit_2."
197
+ unit.cancel(unit_2)
198
+ assert unit.get(unit_2)["state"] == "canceled"
199
+
200
+ #################################################
201
+ # PAUSE/ RESUME/ CANCEL JOB
202
+ #################################################
203
+ say "Pausing job_2."
204
+ job_2.pause
205
+ assert job_2.get["state"] == "paused"
206
+ p "job_2 state: #{job_2.get["state"]}"
207
+
208
+ say "Resuming job_2."
209
+ job_2.resume
210
+ assert job_2.get["state"] == "running"
211
+ p "job_2 state: #{job_2.get["state"]}"
212
+
213
+ say "Canceling job_2."
214
+ job_2.cancel
215
+ assert job_2.get["state"] == "canceled"
216
+ p "job_2 state: #{job_2.get["state"]}"
217
+
218
+ say "Deleting job_1."
219
+ job_1.delete
220
+ assert job_1.get["state"] == "unordered"
221
+
222
+ #################################################
223
+ # JOB LEGEND
224
+ #################################################
225
+ say "Checking job_2 legend."
226
+ assert !job_2.legend.empty?
227
+
228
+ #################################################
229
+ # COPY JOB_2
230
+ #################################################
231
+ say "Copying job_2."
232
+ job_3 = job_2.copy(:all_units => true)
233
+
234
+ say "Updating job_3 title."
235
+ job_3.update(:title => 'Job_3: Copy of Job_2')
236
+ assert job_3.get["title"] == "Job_3: Copy of Job_2"
237
+
238
+ say "-- Waiting for CrowdFlower to process the data."
239
+ wait_until { job_3.get["units_count"] == 14 }
240
+ assert job_3.get["units_count"] == 14
241
+
242
+ #################################################
243
+ # WORKER METHODS
244
+ #################################################
245
+ # Worker tests run against an existing job with an internal CF worker_id number
246
+ say "Starting Worker tests for job #422830."
247
+ job = CrowdFlower::Job.new(422830)
248
+ worker = CrowdFlower::Worker.new(job)
249
+ worker_id = 23542619
250
+
251
+ say "Notifying worker."
252
+ worker.notify(worker_id, "Testing notify method.")
253
+
254
+ say "Bonusing a worker."
255
+ worker.bonus(worker_id, 1, "Here's a bonus for your awesome work!")
256
+
257
+ say "Flagging worker from one of my jobs."
258
+ worker.flag(worker_id, "Testing flag method.", :persist => false)
259
+
260
+ say "Flagging worker from all my jobs."
261
+ worker.flag(worker_id, "Testing flag method across all jobs.", :persist => true)
262
+
263
+ say "Deflagging worker."
264
+ worker.deflag(worker_id, "Testing deflag method.")
265
+
266
+ #################################################
267
+ # JUDGMENT METHODS
268
+ #################################################
269
+ # Judgment tests run against the same job as Worker tests
270
+ say "Starting Judgment tests for job #422830."
271
+ judgment = CrowdFlower::Judgment.new(job)
272
+ unit = CrowdFlower::Unit.new(job)
70
273
 
71
- say "no default api key"
72
- assert_exception_raised(CrowdFlower::UsageError) {CrowdFlower::Job.create("job creation should fail")}
73
- assert_exception_raised(CrowdFlower::UsageError) {job_subclass_with_no_custom_key.create("job creation should fail")}
74
- assert_exception_raised(CrowdFlower::APIError) {job_subclass_with_invalid_custom_key.create("job creation should fail")}
75
- assert job_subclass_with_valid_custom_key.create("should be ok").units.ping['count']
274
+ say "Getting all judgments."
275
+ assert judgment.all.count == job.get["judgments_count"]
76
276
 
77
- say "invalid default api key"
78
- CrowdFlower.connect! "invalid default api key", DOMAIN_BASE
79
- assert_exception_raised(CrowdFlower::APIError) {CrowdFlower::Job.create("job creation should fail")}
80
- assert_exception_raised(CrowdFlower::APIError) {job_subclass_with_no_custom_key.create("job creation should fail")}
81
- assert_exception_raised(CrowdFlower::APIError) {job_subclass_with_invalid_custom_key.create("job creation should fail")}
82
- assert job_subclass_with_valid_custom_key.create("should be ok").units.ping['count']
277
+ say "Checking the judgment number to test judgment.get method."
278
+ assert judgment.get(1243027889)["judgment"] == 1
83
279
 
84
- say "Connecting to the API"
85
- CrowdFlower.connect! API_KEY, DOMAIN_BASE
280
+ say "Checking the number of judgments for a unit to see if a unit's judgment info can be returned."
281
+ assert job.units.judgments(447664267).count == unit.get(447664267)["judgments_count"]
86
282
 
87
- assert CrowdFlower::Job.create("should be ok").units.ping['count']
88
- assert job_subclass_with_no_custom_key.create("should be ok").units.ping['count']
89
- assert job_subclass_with_valid_custom_key.create("should be ok").units.ping['count']
90
- assert_exception_raised(CrowdFlower::APIError) {job_subclass_with_invalid_custom_key.create("job creation should fail")}
91
- # Add this test to check your URL
92
- #assert CrowdFlower::Base.connection.public_url == "localdev.crowdflower.com:80"
93
-
94
- say "Uploading a test CSV"
95
- job = CrowdFlower::Job.upload(File.dirname(__FILE__) + "/sample.csv", "text/csv")
96
-
97
- say "Trying to get all jobs"
98
- assert CrowdFlower::Job.all.first["id"] == job.id
99
-
100
- say "-- Waiting for CrowdFlower to process the data"
101
- wait_until { job.get["units_count"] == 4 }
102
-
103
- say "Adding some more data"
104
- job.upload(File.dirname(__FILE__) + "/sample.csv", "text/csv")
105
-
106
- say "-- Waiting for CrowdFlower to process the data"
107
- # You could also register a webhook to have CrowdFlower notify your
108
- # server.
109
- wait_until { job.get["units_count"] == 8 }
110
-
111
- say "Checking ping."
112
- assert job.units.ping['count'] == 8
113
- assert job.units.ping['done'] == true
114
-
115
- say "Getting the units for this job."
116
- assert job.units.all.size == 8
117
-
118
- say "Checking the status of the job."
119
- assert job.status["tainted_judgments"] == 0
120
-
121
- say "Adding title, instructions, and problem to the job."
122
- job.update({:title => 'testtt',
123
- :instructions => 'testttt fdsf sfds fsdfs fesfsdf',
124
- :cml => '<cml:text label="Text" class="unmodified" validates="required"/>'})
125
-
126
- say "Registering a webhook."
127
- job.update :webhook_uri => "http://localhost:8080/crowdflower"
128
-
129
- say "Tags"
130
- assert job.tags.empty?
131
- job.update_tags ["testing_123", "testing_456"]
132
- assert job.tags.map{|t| t["name"]}.sort == ["testing_123", "testing_456"]
133
- job.remove_tags ["testing_123"]
134
- assert job.tags.map{|t| t["name"]} == ["testing_456"]
135
- job.add_tags ["testing_789"]
136
- assert job.tags.map{|t| t["name"]} == ["testing_456", "testing_789"]
137
-
138
- say "Copying the existing job to a new one."
139
- job2 = job.copy :all_units => true
140
-
141
- say "-- Waiting for CrowdFlower to finish copying the job."
142
- # You could also register a webhook to have CrowdFlower notify your
143
- # server.
144
- wait_until { job2.get["units_count"] == 8 }
145
-
146
- say "Ordering the job."
147
- order = CrowdFlower::Order.new(job)
148
- unit_count = 8
149
- order.debit(8)
150
- wait_until { job.get["state"].casecmp('running') == 0}
151
-
152
- say "Checking enabled_channels."
153
- assert !job.channels["enabled_channels"].empty?
154
- # To see exact list of enabled jobs:
155
- # p job.channels["enabled_channels"]
156
-
157
- say "Order job2 to check channels when specifying channel parameter."
158
- order = CrowdFlower::Order.new(job2)
159
- order.debit(8, "channel"=>"cf_internal")
160
- assert job2.channels["enabled_channels"] == ["cf_internal"]
161
-
162
- say "Canceling the unit."
163
- unit_id = job.units.all.to_a[0][0]
164
- unit = CrowdFlower::Unit.new(job)
165
- wait_until { unit.get(unit_id)['state'] == 'judgable' }
166
- puts unit.cancel(unit_id).inspect
167
- assert unit.get(unit_id)['state'] == 'canceled'
283
+ #################################################
284
+ # DOWNLOAD REPORTS - missing assetions
285
+ #################################################
286
+ # These run against the completed job from readme examples; they print status codes as they go
287
+ say "Startiing report tests for job #418404."
288
+ job = CrowdFlower::Job.new(418404)
289
+
290
+ say "Downloading Full CSV"
291
+ job.download_csv(:full, "full_report.zip")
292
+
293
+ say "Downloading Aggregated CSV"
294
+ job.download_csv(:aggregated, "agg_report.zip")
295
+
296
+ say "Downloading Source CSV"
297
+ job.download_csv(:source, "source_report.zip")
298
+
299
+ say "Downloading Test Questions CSV"
300
+ job.download_csv(:gold_report, "gold_report.zip")
301
+
302
+ say "Downloading Worker CSV"
303
+ job.download_csv(:workset, "workset_report.zip")
168
304
 
169
- say "Webhook test needs to be written."
170
- #job.test_webhook
305
+ say "Downloading JSON"
306
+ job.download_csv(:json, "json_report.zip")
171
307
 
308
+ #################################################
309
+ # END OF TESTS
310
+ #################################################
172
311
  say ">-< Tests complete. >-<"
metadata CHANGED
@@ -1,8 +1,7 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: crowdflower
3
3
  version: !ruby/object:Gem::Version
4
- version: 0.11.0
5
- prerelease:
4
+ version: 0.12.0
6
5
  platform: ruby
7
6
  authors:
8
7
  - Brian P O'Rourke
@@ -10,31 +9,27 @@ authors:
10
9
  autorequire:
11
10
  bindir: bin
12
11
  cert_chain: []
13
- date: 2014-05-02 00:00:00.000000000 Z
12
+ date: 2014-05-05 00:00:00.000000000 Z
14
13
  dependencies:
15
14
  - !ruby/object:Gem::Dependency
16
15
  name: httparty
17
16
  requirement: !ruby/object:Gem::Requirement
18
- none: false
19
17
  requirements:
20
- - - ! '>='
18
+ - - ">="
21
19
  - !ruby/object:Gem::Version
22
20
  version: 0.7.4
23
21
  type: :runtime
24
22
  prerelease: false
25
23
  version_requirements: !ruby/object:Gem::Requirement
26
- none: false
27
24
  requirements:
28
- - - ! '>='
25
+ - - ">="
29
26
  - !ruby/object:Gem::Version
30
27
  version: 0.7.4
31
- description: ! 'A toolkit for interacting with CrowdFlower via the REST API.
32
-
28
+ description: |+
29
+ A toolkit for interacting with CrowdFlower via the REST API.
33
30
 
34
31
  This is alpha software. Have fun!
35
32
 
36
-
37
- '
38
33
  email: brian@crowdflower.com
39
34
  executables: []
40
35
  extensions: []
@@ -42,7 +37,7 @@ extra_rdoc_files:
42
37
  - LICENSE
43
38
  - README.md
44
39
  files:
45
- - .document
40
+ - ".document"
46
41
  - CONTRIBUTORS
47
42
  - HISTORY.md
48
43
  - LICENSE
@@ -51,6 +46,7 @@ files:
51
46
  - VERSION
52
47
  - bindev/cl_skel.rb
53
48
  - bindev/crowdflower.rb
49
+ - config/account_info.sample.yml
54
50
  - crowdflower.gemspec
55
51
  - lib/crowdflower.rb
56
52
  - lib/crowdflower/base.rb
@@ -59,30 +55,29 @@ files:
59
55
  - lib/crowdflower/order.rb
60
56
  - lib/crowdflower/unit.rb
61
57
  - lib/crowdflower/worker.rb
58
+ - test/crowdshopping.csv
62
59
  - test/integration_tests.rb
63
- - test/sample.csv
64
60
  homepage: http://github.com/dolores/ruby-crowdflower
65
61
  licenses: []
62
+ metadata: {}
66
63
  post_install_message:
67
64
  rdoc_options: []
68
65
  require_paths:
69
66
  - lib
70
67
  required_ruby_version: !ruby/object:Gem::Requirement
71
- none: false
72
68
  requirements:
73
- - - ! '>='
69
+ - - ">="
74
70
  - !ruby/object:Gem::Version
75
71
  version: '0'
76
72
  required_rubygems_version: !ruby/object:Gem::Requirement
77
- none: false
78
73
  requirements:
79
- - - ! '>='
74
+ - - ">="
80
75
  - !ruby/object:Gem::Version
81
76
  version: '0'
82
77
  requirements: []
83
78
  rubyforge_project:
84
- rubygems_version: 1.8.23
79
+ rubygems_version: 2.2.0
85
80
  signing_key:
86
- specification_version: 3
81
+ specification_version: 4
87
82
  summary: a toolkit for the CrowdFlower API
88
83
  test_files: []
@@ -1,5 +0,0 @@
1
- "created_at","from_user","text"
2
- "Sat, 10 Oct 2009 01:00:07 +0000","sogowave","CrowdFlower easily connects you with thousands of people online, around the clock. http://ow.ly/rpaT"
3
- "Sat, 10 Oct 2009 01:00:05 +0000","SogoPR","CrowdFlower easily connects you with thousands of people online, around the clock. http://ow.ly/rpaS"
4
- "Fri, 09 Oct 2009 15:00:07 +0000","SIIA_Software","...and don't forget: @CloudTrigger @businessobjects @longjump @CrowdFlower http://bit.ly/397y72"
5
- "Fri, 02 Oct 2009 19:49:50 +0000","julzie","RT @rev2 CrowdFlower - Labor as a Service in a New Concept | Rev2.org http://retwt.me/v80H - GO LaaS!!"