crowdflower 0.11.0 → 0.12.0
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +7 -0
- data/README.md +62 -40
- data/VERSION +1 -1
- data/config/account_info.sample.yml +5 -0
- data/crowdflower.gemspec +9 -7
- data/lib/crowdflower/job.rb +4 -0
- data/test/crowdshopping.csv +7 -0
- data/test/integration_tests.rb +249 -110
- metadata +14 -19
- data/test/sample.csv +0 -5
checksums.yaml
ADDED
@@ -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
|
-
#####[
|
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
|
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
|
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
|
152
|
+
#####CHANNELS: http://api.crowdflower.com/v1/jobs/418404/channels
|
152
153
|
|
153
154
|
```ruby
|
154
|
-
|
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("
|
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
|
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
|
-
|
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:
|
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:
|
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
|
-
|
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:
|
279
|
+
#####PAUSE: Can only be called on running jobs.
|
267
280
|
|
268
281
|
```ruby
|
269
282
|
job.pause
|
270
283
|
```
|
271
284
|
|
272
|
-
#####RESUME:
|
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:
|
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
|
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(
|
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:
|
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(
|
327
|
+
worker.reject(99999999)
|
315
328
|
```
|
316
329
|
|
317
|
-
#####WORKER.NOTIFY: Sends a
|
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,
|
321
|
-
worker.notify(
|
333
|
+
worker.notify(worker_id, message)
|
334
|
+
worker.notify(99999999, "Thanks for working on this task!")
|
322
335
|
```
|
323
336
|
|
324
|
-
#####WORKER.FLAG:
|
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(
|
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:
|
348
|
+
#####WORKER.DEFLAG: Removes flag and allows contributor to continue work.
|
332
349
|
```ruby
|
333
|
-
worker.deflag(worker_id)
|
334
|
-
worker.deflag(
|
350
|
+
worker.deflag(worker_id, reason)
|
351
|
+
worker.deflag(99999999, "Worker was mistakenly flagged.")
|
335
352
|
```
|
336
353
|
|
337
|
-
#####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(
|
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(
|
364
|
+
job.units.judgments(9999999999)
|
348
365
|
```
|
349
366
|
|
350
|
-
#####LEGEND: Returns all
|
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:
|
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:
|
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
|
375
|
-
job.download_csv(full,
|
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.
|
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
|
data/crowdflower.gemspec
CHANGED
@@ -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.
|
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-
|
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/
|
38
|
-
"test/
|
40
|
+
"test/crowdshopping.csv",
|
41
|
+
"test/integration_tests.rb"
|
39
42
|
]
|
40
43
|
s.homepage = "http://github.com/dolores/ruby-crowdflower"
|
41
|
-
s.
|
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 =
|
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"])
|
data/lib/crowdflower/job.rb
CHANGED
@@ -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,"",,
|
data/test/integration_tests.rb
CHANGED
@@ -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
|
-
|
8
|
-
|
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
|
-
|
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
|
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
|
-
|
67
|
-
|
68
|
-
|
69
|
-
|
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 "
|
72
|
-
|
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 "
|
78
|
-
|
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 "
|
85
|
-
|
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
|
-
|
88
|
-
|
89
|
-
|
90
|
-
|
91
|
-
|
92
|
-
|
93
|
-
|
94
|
-
say "
|
95
|
-
job
|
96
|
-
|
97
|
-
say "
|
98
|
-
|
99
|
-
|
100
|
-
say "
|
101
|
-
|
102
|
-
|
103
|
-
say "
|
104
|
-
job.
|
105
|
-
|
106
|
-
say "
|
107
|
-
|
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 "
|
170
|
-
|
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.
|
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-
|
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:
|
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:
|
79
|
+
rubygems_version: 2.2.0
|
85
80
|
signing_key:
|
86
|
-
specification_version:
|
81
|
+
specification_version: 4
|
87
82
|
summary: a toolkit for the CrowdFlower API
|
88
83
|
test_files: []
|
data/test/sample.csv
DELETED
@@ -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!!"
|