firstjob 1.0.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.
@@ -0,0 +1,49 @@
1
+ module Firstjob
2
+ class HttpParser
3
+ def self.parse_json_to_hash(json, hash)
4
+ json.each{|object| hash[object["id"]] ? hash[object["id"]].merge!(object) : hash[object["id"]] = object}
5
+ return hash
6
+ end
7
+
8
+ def self.parse_response(response)
9
+ Firstjob.last_response = response
10
+ Firstjob.last_request = response.request
11
+ case response.code
12
+ when 200..201
13
+ # "All good!"
14
+ return response.body
15
+ when 401
16
+ Firstjob.invalidate_access_token!
17
+ raise "Error 401: Unauthorized. Check login info.\n #{response.body}"
18
+ when 403
19
+ raise "Error 403: Forbidden"
20
+ when 404
21
+ raise "Error 404 not found"
22
+ when 500...600
23
+ raise "ZOMG ERROR #{response.code}: #{response.request.path}, #{response.body}"
24
+ else
25
+ raise "Error #{response.code}, unkown response: #{response.request.path}, #{response.body}"
26
+ end
27
+ end
28
+ def self.parse_json_response(response)
29
+ Firstjob.last_response = response
30
+ Firstjob.last_request = response.request
31
+ case response.code
32
+ when 200..201
33
+ # "All good!"
34
+ return JSON.parse(response.body)
35
+ when 401
36
+ Firstjob.invalidate_access_token!
37
+ raise "Error 401: Unauthorized. Check login info.\n #{response.body}"
38
+ when 403
39
+ raise "Error 403: Forbidden"
40
+ when 404
41
+ raise "Error 404 not found"
42
+ when 500...600
43
+ raise "ZOMG ERROR #{response.code}: #{response.request.path}, #{response.body}"
44
+ else
45
+ raise "Error #{response.code}, unkown response: #{response.request.path}, #{response.body}"
46
+ end
47
+ end
48
+ end
49
+ end
@@ -0,0 +1,15 @@
1
+ module Firstjob
2
+ class LookingFor
3
+ LOOKING_FORS = YAML.load_file("lib/data/looking_fors.yaml")["looking_fors"].symbolize_keys!
4
+
5
+ # Poor's man all
6
+ def self.all
7
+ LOOKING_FORS
8
+ end
9
+
10
+ # Poor's man find
11
+ def self.find(id)
12
+ LOOKING_FORS[id]
13
+ end
14
+ end
15
+ end
@@ -0,0 +1,97 @@
1
+ module Firstjob
2
+ class Publication
3
+ attr_accessor :title, :description, :career_status_id,
4
+ :looking_for_id, :location, :years_experience, :salary,
5
+ :english_level_id, :excel_level_id, :careers_ids,
6
+ :universities_ids, :id, :slug, :status, :postulantes
7
+
8
+ def initialize(attributes={})
9
+ # Default values
10
+ @salary = 0
11
+ @location = "Chile"
12
+ @career_status_id = 1
13
+ @years_experience = 0
14
+ @english_level_id = 1
15
+ @excel_level_id = 1
16
+
17
+ attributes.each do |key, value|
18
+ self.send("#{key}=", value)
19
+ end
20
+ end
21
+
22
+ def body
23
+ default_publication_json()
24
+ end
25
+
26
+ def self.create(params)
27
+ publication = new(params)
28
+ response = publication.publish()
29
+ end
30
+
31
+ def publish
32
+ # Post publication
33
+ response = Firstjob.post("/api/publish_job",
34
+ Firstjob.options.merge(
35
+ body: Firstjob.body.merge({publish_job: self.body}).to_json
36
+ )
37
+ )
38
+ response_body = HttpParser.parse_json_response(response)
39
+
40
+ # load results
41
+ @slug = response_body["slug"]
42
+ @id = response_body["id"]
43
+ @status = response_body["status"]
44
+ return self
45
+ end
46
+
47
+ def url
48
+ "#{Firstjob.base_uri}/jobs/#{slug}"
49
+ end
50
+
51
+ def postulants
52
+ # Get postulants
53
+ response = Firstjob.post("/api/get_applicants_job",
54
+ Firstjob.options.merge(
55
+ body: Firstjob.body.merge({job_id: @id}).to_json
56
+ )
57
+ )
58
+ response_body = HttpParser.parse_json_response(response)
59
+
60
+ # load results
61
+ @postulantes = response_body["postulantes"]
62
+ return @postulantes
63
+ end
64
+
65
+ def destroy
66
+ response = Firstjob.post("/api/doregister_job",
67
+ Firstjob.options.merge(
68
+ body: Firstjob.body.merge({job_id: id}).to_json
69
+ )
70
+ )
71
+ response_body = HttpParser.parse_response(response)
72
+ end
73
+
74
+ private
75
+ def default_publication_json
76
+ hash = {
77
+ title: title, # required
78
+ description: description, # required, max 300 caracters
79
+ career_status: career_status_id, # required
80
+ looking_for: looking_for_id, # required
81
+ location: location, # required
82
+ years_experience: years_experience, # required, number
83
+ salary: salary, # required, number. Can be 0
84
+ english_level: english_level_id, # required, number
85
+ excel_level: excel_level_id, # required, number
86
+ }
87
+ if careers_ids.present?
88
+ hash[:careers] = careers_ids # optional, array or NOT PRESENT
89
+ end
90
+ if universities_ids.present?
91
+ hash[:universities] = universities_ids # optional, array or NOT PRESENT
92
+ end
93
+ return hash
94
+ end
95
+
96
+ end
97
+ end
@@ -0,0 +1,15 @@
1
+ module Firstjob
2
+ class University
3
+ UNIVERSITIES = YAML.load_file("lib/data/universities.yaml")["universities"].symbolize_keys!
4
+
5
+ # Poor's man all
6
+ def self.all
7
+ UNIVERSITIES
8
+ end
9
+
10
+ # Poor's man find
11
+ def self.find(id)
12
+ UNIVERSITIES[id]
13
+ end
14
+ end
15
+ end
@@ -0,0 +1,3 @@
1
+ module Firstjob
2
+ VERSION = "1.0.0"
3
+ end
data/lib/firstjob.rb ADDED
@@ -0,0 +1,142 @@
1
+ require 'rails'
2
+ require 'time'
3
+ require 'httparty'
4
+ require 'erb'
5
+
6
+ require 'firstjob/university'
7
+ require 'firstjob/career'
8
+ require 'firstjob/career_status'
9
+ require 'firstjob/english_level'
10
+ require 'firstjob/excel_level'
11
+ require 'firstjob/looking_for'
12
+ require 'firstjob/university'
13
+ require 'firstjob/publication'
14
+ require 'firstjob/http_parser'
15
+
16
+ module Firstjob
17
+ include HTTParty
18
+ base_uri 'https://api.firstjob.com'
19
+
20
+ # API login configuration, need initialization setup to work
21
+ mattr_accessor :grant_type
22
+ @@grant_type = "password"
23
+
24
+ mattr_accessor :client_id
25
+ @@client_id = nil
26
+
27
+ mattr_accessor :username
28
+ @@username = nil
29
+
30
+ mattr_accessor :email
31
+ @@email = nil
32
+
33
+ mattr_accessor :password
34
+ @@password = nil
35
+
36
+ mattr_accessor :access_token
37
+ @@access_token = nil
38
+
39
+ @@token_type= nil
40
+
41
+ mattr_accessor :expires_in
42
+ @@expires_in = nil
43
+
44
+ mattr_accessor :access_token_updated_at
45
+ @@access_token_updated_at = nil
46
+
47
+ mattr_accessor :options
48
+ @@options = nil
49
+
50
+ mattr_accessor :body
51
+ @@body = nil
52
+
53
+ mattr_accessor :last_request
54
+ mattr_accessor :last_response
55
+ @@last_request = nil
56
+ @@last_response = nil
57
+
58
+ @@try_counter = 0
59
+
60
+
61
+ # Default way to setup Firstjob.
62
+ def self.setup
63
+ yield self
64
+ # It does not use basic http auth, it passes the username and password in the body of the request...
65
+ #@@options = {headers: { "Accept" => "application/json", "Content-Type" => "application/json"}, basic_auth: {username: @@username, password: @@password}}
66
+ @@options = {headers: { "Accept" => "application/json", "Content-Type" => "application/json"}}
67
+ @@body = {username: @@username, password: @@password}
68
+ end
69
+
70
+ # Publicaciones
71
+ # creates and publish a publication
72
+ def self.publish(params)
73
+ publication = Firstjob::Publication.create(params)
74
+ return publication
75
+ end
76
+
77
+ def self.update_publication(publication_id, json)
78
+ update_publication_path = "/v0/empresas/avisos/#{publication_id}"
79
+ response = self.post(update_publication_path, @@options.merge(body: json, headers: { "Accept" => "application/json", "Content-Type" => "application/json"}))
80
+
81
+ if HttpParser.parse_response(response)
82
+ case response.code
83
+ when 201
84
+ # "Publication updated, All good!"
85
+ return response # body contains id del proceso publicado
86
+ when 200
87
+ # "TODO: Uhm.. no idea, is this good?"
88
+ return response # body contains id del proceso publicado?
89
+ end
90
+ end
91
+ end
92
+
93
+ def self.publish_publication(publication_id, pais_id, plan_publication_id)
94
+ publish_publication_path = "/v0/empresas/avisos/#{publication_id}/publicacion/#{plan_publication_id}"
95
+ response = self.put(publish_publication_path, @@options.merge(query: @@options[:query].merge({paisId: pais_id})))
96
+
97
+ if HttpParser.parse_response(response)
98
+ case response.code
99
+ when 201
100
+ # "Publication created, All good!"
101
+ return response # body contains id del proceso publicado
102
+ when 200
103
+ # "TODO: Uhm.. no idea, is this good?"
104
+ return response # body contains id del proceso publicado?
105
+ end
106
+ end
107
+ end
108
+
109
+ def self.get_publication_url(publication_id)
110
+ "http://www.laborum.cl/empleos/#{publication_id}.html"
111
+ end
112
+
113
+ def self.get_publication(publication_id)
114
+ get_publication_path = "/v0/empresas/avisos/#{publication_id}"
115
+ response = self.get(get_publication_path, @@options)
116
+
117
+ return HttpParser.parse_response_to_json(response)
118
+ end
119
+
120
+ def self.get_postulations_in_publication(publication_id, page=0, postulations_per_page=20)
121
+ get_postulations_in_publication_path = "/v0/empresas/avisos/#{publication_id}/postulaciones"
122
+ response = self.get(get_postulations_in_publication_path, @@options.merge(query: @@options[:query].merge({page: page, perPage: postulations_per_page})))
123
+
124
+ return HttpParser.parse_response_to_json(response)
125
+ end
126
+
127
+ def self.get_postulation(postulation_id)
128
+ get_postulation_path = "/v0/empresas/postulaciones/#{postulation_id}"
129
+ response = self.get(get_postulation_path, @@options)
130
+
131
+ return HttpParser.parse_response_to_json(response)
132
+ end
133
+
134
+ def self.destroy_publication(publication_id)
135
+ destroy_publication_path = "/v0/empresas/avisos/#{publication_id}"
136
+ response = self.delete(destroy_publication_path, @@options)
137
+
138
+ return HttpParser.parse_response(response)
139
+ end
140
+
141
+ end
142
+
@@ -0,0 +1,5 @@
1
+ Firstjob.setup do |config|
2
+ config.username = ENV["FIRSTJOB_USERNAME"]
3
+ config.password = ENV["FIRSTJOB_PASSWORD"]
4
+ config.base_uri ENV["FIRSTJOB_BASE_URI"]
5
+ end
@@ -0,0 +1,69 @@
1
+ class FirstjobFixture
2
+ def self.publication
3
+ publication = Firstjob::Publication.new
4
+ publication.body[:descripcion] = "Descripcion de la publicacion de prueba" # required
5
+ publication.body[:titulo] = "Publicacion de prueba" # required
6
+ #publication.body[:referencia] = "" #optional
7
+
8
+ # We can use an id for denominacion
9
+ publication.body[:denominacion][:id] = 0 # required
10
+ # or we can create a new one
11
+ #publication.body[:denominacion][:nombre] = ""
12
+ #publication.body[:denominacion][:logo] = ""
13
+
14
+ #publication.body[:preguntas] # optional
15
+ publication.body[:preguntas][0][:simple][:texto] = "Pregunta de prueba" # optional
16
+ publication.body[:preguntas] << {choice: {texto: "Pregunta de alternativas", indiceCorrecta: 2, opciones: [{opcion: "respuesta 1"}, {opcion: "respuesta 2"}, {opcion: "respuesta 3"}, {opcion: "respuesta 4"}]}} # optional
17
+
18
+ #publication.body["postulantesDiscapacitados"] false, #optional
19
+
20
+ #publication.body["lugarTrabajo"]
21
+ #publication.body[:lugarTrabajo][:id] = 0 # TODO: found in the developers site but not in the documentation
22
+ publication.body[:lugarTrabajo][:paisId] = 1 # required
23
+ publication.body[:lugarTrabajo][:zonaId] = 18 # required
24
+ publication.body[:lugarTrabajo][:localidadId] = 6050 # required
25
+ publication.body[:lugarTrabajo][:direccion] = "San Martin 256" # required
26
+ publication.body[:lugarTrabajo][:mostrarDireccionEnAviso] = false # required
27
+
28
+ publication.body[:recepcionCandidato][:electronica][:email] = "test@domain.com" # required
29
+ # recepcionCandidato can aslo be with dias, rangoHorario and direccion
30
+ publication.body[:areaId] = 19 # required
31
+ publication.body[:subAreaId] = 26 # required
32
+ ####publication.body[:requisitos] # optional
33
+ ###publication.body[:requisitos][:experiencia]
34
+ publication.body[:requisitos][:experiencia][:minimo] = 60 # required
35
+ publication.body[:requisitos][:experiencia][:excluyente] = false # required
36
+ ###publication.body[:requisitos][:edad]
37
+ publication.body[:requisitos][:edad][:edadMinima] = 18 # required
38
+ publication.body[:requisitos][:edad][:edadMaxima] = 40 # required
39
+ publication.body[:requisitos][:edad][:excluyente] = false # required
40
+ ###publication.body["requisitos"]["educacion"]
41
+ publication.body[:requisitos][:educacion][:estadoEstudioId] = 2 # required
42
+ publication.body[:requisitos][:educacion][:tipoEstudioId] = 3 # required
43
+ publication.body[:requisitos][:educacion][:excluyente] = false # required
44
+ ###publication.body["requisitos"]["idiomas"]
45
+ publication.body[:requisitos][:idiomas][0][:nivelId] = 10 # required
46
+ publication.body[:requisitos][:idiomas][0][:idiomaId] = 1 # required
47
+ publication.body[:requisitos][:idiomas][0][:excluyente] = false # required
48
+ ###publication.body["requisitos"]["residencia"]
49
+ publication.body[:requisitos][:residencia][:cercania] = "provincia" # required
50
+ #publication.body[:requisitos][:residencia][:cantidadKm] = 0 # optional
51
+ publication.body[:requisitos][:residencia][:excluyente] = false # required
52
+ ###publication.body["requisitos"]["salario"]
53
+ publication.body[:requisitos][:salario][:tipo] = "bruto" # required
54
+ publication.body[:requisitos][:salario][:salarioMinimo] = 200000 # required
55
+ publication.body[:requisitos][:salario][:salarioMaximo] = 1000000 # required
56
+ publication.body[:requisitos][:salario][:frecuenciaPagoId] = 4 # required
57
+ publication.body[:requisitos][:salario][:mostrarAviso] = true # required
58
+ publication.body[:requisitos][:salario][:solicitarCandidato] = true # required
59
+ publication.body[:requisitos][:salario][:excluyente] = false # optional
60
+ ###publication.body[:requisitos][:genero]
61
+ publication.body[:requisitos][:genero][:nombre] = "masculino" # required
62
+ publication.body[:requisitos][:genero][:excluyente] = false # required
63
+ publication.body[:tipoTrabajoId] = 4 # required
64
+
65
+ return publication
66
+
67
+ # publish(publication.body.to_publication.body)
68
+ end
69
+ end
@@ -0,0 +1,174 @@
1
+ require 'spec_helper'
2
+
3
+ describe Bumeran do
4
+ it 'is a sanity test' do
5
+ end
6
+
7
+ it 'has a valid model' do
8
+ Bumeran.has_valid_access_token?.should be_true
9
+ end
10
+
11
+ it 'can create publication', create: true do
12
+ publication = BumeranFixture.publication
13
+ result = Bumeran.create_aviso(publication.body.to_json)
14
+ pp result
15
+ result.to_i.should > 0
16
+ end
17
+
18
+ it 'can publish publication', create: true do
19
+ publication = BumeranFixture.publication
20
+ published_publication_id = Bumeran.publish(publication.body.to_json, 1, 30) # pais Argentina = 1, plan publicacion "simple" = 30
21
+ pp published_publication_id
22
+ published_publication_id.to_i.should > 0
23
+ end
24
+
25
+ it 'can create, get, publish and destroy publication', publish: true do
26
+ publication = BumeranFixture.publication
27
+ publication_id = Bumeran.create_aviso(publication.body.to_json)
28
+ pp publication_id
29
+ publication_id.should > 0
30
+
31
+ publication = Bumeran.get_publication(publication_id)
32
+ pp publication
33
+ publication["id"].should > 0
34
+
35
+ published_publication = Bumeran.publicar_aviso(publication_id, 1, 30) # pais Argentina = 1, plan publicacion "simple" = 30
36
+ pp published_publication
37
+
38
+ deleted_publication = Bumeran.destroy_aviso(publication_id)
39
+ pp deleted_publication
40
+ end
41
+
42
+ it 'can get areas', getters: true do
43
+ pp Bumeran.areas
44
+ Bumeran.areas.count.should > 0
45
+ end
46
+
47
+ it 'can get subareas', getters: true do
48
+ pp Bumeran.subareas
49
+ Bumeran.subareas.count.should > 0
50
+ end
51
+
52
+ it 'can get frecuencias_pago', getters: true do
53
+ pp Bumeran.frecuencias_pago
54
+ Bumeran.frecuencias_pago.count.should > 0
55
+ end
56
+
57
+ it 'can get paises' do
58
+ pp Bumeran.paises
59
+ Bumeran.paises.count.should > 0
60
+ end
61
+
62
+ # localidades gives error! (API problem)
63
+ it 'can get paises, zonas, localidades, and plan plublicaciones' do
64
+ pp Bumeran.paises
65
+ Bumeran.paises.count.should > 0
66
+ Bumeran.zonas.count.should > 0
67
+ #Bumeran.localidades.count.should > 0
68
+ Bumeran.plan_publicaciones.count.should > 0
69
+ end
70
+
71
+ it 'can get denominaciones', getters: true do
72
+ pp Bumeran.denominaciones
73
+ Bumeran.denominaciones.count.should > 0
74
+ end
75
+
76
+ it 'can get direcciones', getters: true do
77
+ pp Bumeran.direcciones
78
+ Bumeran.direcciones.count.should > 0
79
+ end
80
+
81
+ it 'can get frecuencias_pago', getters: true do
82
+ pp Bumeran.frecuencias_pago
83
+ Bumeran.frecuencias_pago.count.should > 0
84
+ end
85
+
86
+
87
+ it 'can get idiomas', getters: true do
88
+ pp Bumeran.idiomas
89
+ Bumeran.idiomas.count.should > 0
90
+ end
91
+
92
+ it 'can get industrias', getters: true do
93
+ pp Bumeran.industrias
94
+ Bumeran.industrias.count.should > 0
95
+ end
96
+
97
+ it 'can get niveles_idiomas', getters: true do
98
+ pp Bumeran.niveles_idiomas
99
+ Bumeran.niveles_idiomas.count.should > 0
100
+ end
101
+
102
+ it 'can get tipos_trabajo', getters: true do
103
+ pp Bumeran.tipos_trabajo
104
+ Bumeran.tipos_trabajo.count.should > 0
105
+ end
106
+
107
+ it 'can get areas_estudio', getters: true do
108
+ pp Bumeran.areas_estudio
109
+ Bumeran.areas_estudio.count.should > 0
110
+ end
111
+
112
+ it 'can get estados_estudio', getters: true do
113
+ pp Bumeran.estados_estudio
114
+ Bumeran.estados_estudio.count.should > 0
115
+ end
116
+
117
+ it 'can get tipos_estudio', getters: true do
118
+ pp Bumeran.tipos_estudio
119
+ Bumeran.tipos_estudio.count.should > 0
120
+ end
121
+
122
+ it 'can get tipos_estudio', getters: true do
123
+ pp Bumeran.tipos_estudio
124
+ Bumeran.tipos_estudio.count.should > 0
125
+ end
126
+
127
+ it 'can get estudio', getters: true do
128
+ params = {}
129
+ params['curriculum_id']=1000
130
+ estudio = Bumeran.get_estudio(1285190, params)
131
+ pp estudio
132
+ estudio["titulo"].class.should be String
133
+ end
134
+
135
+ it 'can get conocimiento', getters: true do
136
+ #conocimiento = Bumeran.get_conocimiento(140)
137
+ #pp conocimiento
138
+ #conocimiento["nombre"].class.should be String
139
+ params = {}
140
+ params['curriculum_id']=1000
141
+ conocimiento = Bumeran.get_conocimiento(140, params)
142
+ pp conocimiento
143
+ conocimiento["nombre"].class.should be String
144
+ end
145
+
146
+ it 'can get conocimiento_custom', getters: true do
147
+ params = {}
148
+ params['curriculum_id']=1000
149
+ conocimiento_custom = Bumeran.get_conocimiento_custom(623710, params)
150
+ pp conocimiento_custom
151
+ conocimiento_custom["nombre"].class.should be String
152
+ end
153
+
154
+ it 'can get experiencia_laboral', getters: true do
155
+ params = {}
156
+ params['curriculum_id']=1000
157
+ experiencia_laboral = Bumeran.get_experiencia_laboral(1651500, params)
158
+ pp experiencia_laboral
159
+ experiencia_laboral["puesto"].class.should be String
160
+ end
161
+
162
+ it 'can get postulacion', getters: true do
163
+ postulacion = Bumeran.get_postulacion(1000)
164
+ pp postulacion
165
+ postulacion["estado"].class.should be String
166
+ end
167
+
168
+ it 'can get curriculum', getters: true do
169
+ curriculum = Bumeran.get_curriculum(1000)
170
+ pp curriculum
171
+ curriculum["nombre"].class.should be String
172
+ end
173
+
174
+ end
@@ -0,0 +1,29 @@
1
+ # This file was generated by the `rspec --init` command. Conventionally, all
2
+ # specs live under a `spec` directory, which RSpec adds to the `$LOAD_PATH`.
3
+ # Require this file using `require "spec_helper"` to ensure that it is only
4
+ # loaded once.
5
+
6
+ #
7
+ # See http://rubydoc.info/gems/rspec-core/RSpec/Core/Configuration
8
+ RSpec.configure do |config|
9
+ config.treat_symbols_as_metadata_keys_with_true_values = true
10
+ config.run_all_when_everything_filtered = true
11
+ config.filter_run focus: true
12
+
13
+ Dir["./spec/fixtures/*.rb"].sort.each { |f| require f }
14
+
15
+ # Run specs in random order to surface order dependencies. If you find an
16
+ # order dependency and want to debug it, you can fix the order by providing
17
+ # the seed, which is printed after each run.
18
+ # --seed 1234
19
+ config.order = 'random'
20
+ end
21
+
22
+ require 'firstjob'
23
+ require 'firstjob/publication'
24
+ begin
25
+ require 'firstjob_initializer_helper'
26
+ rescue Exception => e
27
+ raise "firstjob_initializer_helper does't exist, follow the instructions in the README"
28
+ end
29
+ require 'pry-byebug'