peoplegroup-connectors 0.1.8 → 0.1.13

Sign up to get free protection for your applications and to get access to all the features.
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 970e02a20813047a8a3673fc205db913c74a05f796c638527d646a47a45b6b3c
4
- data.tar.gz: bface6e5766fd959d452b59a50b329e0382a551657652048a170219193903106
3
+ metadata.gz: 82e25996e920cfe64593a348342c127fa24a668edd0b3e9d42bfe3306c235ea1
4
+ data.tar.gz: edf8d6e9f82c5aac0523ec56070d2e4f7d24d07c846f5f05fd85e57daf8a8bec
5
5
  SHA512:
6
- metadata.gz: 953986438996dd78ee42d3a8b38530d17552225f7f32bc13155aa9fec254243fe7202fa3f5ad40559988f8b1812e5a85f9fa99585dbdb147c9713043db9d6d3f
7
- data.tar.gz: 40e562e07ac901a4557680933f6ac0bce5eebe9f8ab9f0d40289a46e96a875fd0b27355bc5a386a3988ae838bbf9c06dcd767a78cbb704f0dc9e2697f520a509
6
+ metadata.gz: a1bad958065dfb45ac92dbeb760abc7078f057fef82dd90fce95203a2f08a09a98f1bf5c403ec4b928fddb1f1b6b1ab37e6eeeda13dd6f59d947dceee1071064
7
+ data.tar.gz: 30affe0fb75c7877c6cdfde690ddcaeb30193603971f8aab4824c88f6bad87bbb1f9075640a40a352be3049c2ce30dedd97adea97ba3f2c4cd0e75244f304794
data/README.md CHANGED
@@ -1,8 +1,6 @@
1
1
  # PeopleGroup::Connectors
2
2
 
3
- Welcome to your new gem! In this directory, you'll find the files you need to be able to package up your Ruby library into a gem. Put your Ruby code in the file `lib/peoplegroup/connectors`. To experiment with that code, run `bin/console` for an interactive prompt.
4
-
5
- TODO: Delete this and the text above, and describe your gem
3
+ The gem contains wrapper code for all the services we sync with. The reason we bundled this in a separate gem is to not have duplicate code in our different projects.
6
4
 
7
5
  ## Installation
8
6
 
@@ -22,7 +20,10 @@ Or install it yourself as:
22
20
 
23
21
  ## Usage
24
22
 
25
- TODO: Write usage instructions here
23
+ ```ruby
24
+ gitlab_client = PeopleGroup::Connectors::GitLab.new
25
+ gitlab_client.find_gitlabber(:email, 'some-email@domain.com')
26
+ ```
26
27
 
27
28
  ## Development
28
29
 
@@ -4,6 +4,8 @@ require_relative 'connectors/version'
4
4
  require_relative 'connectors/gitlab'
5
5
  require_relative 'connectors/slack'
6
6
  require_relative 'connectors/pto_roots'
7
+ require_relative 'connectors/bamboo'
8
+ require_relative 'connectors/greenhouse'
7
9
 
8
10
  module PeopleGroup
9
11
  module Connectors
@@ -0,0 +1,175 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'bamboozled'
4
+
5
+ module PeopleGroup
6
+ module Connectors
7
+ class Bamboo
8
+ MAX_RETRIES = 3
9
+
10
+ def initialize
11
+ @client = Bamboozled.client(subdomain: 'gitlab', api_key: ENV['BAMBOO_API_KEY'])
12
+ end
13
+
14
+ def get_employee_details(id)
15
+ employees.find { |emp| emp['id'] == id.to_s }
16
+ end
17
+
18
+ def get_employee(id)
19
+ @client.employee.find(id, %w[firstName lastName jobTitle supervisor hireDate country location department division workEmail customCostCenter])
20
+ end
21
+
22
+ def search_employee(name)
23
+ return if name.empty?
24
+
25
+ employees.find do |emp|
26
+ [
27
+ emp['displayName']&.downcase,
28
+ "#{emp['firstName']&.downcase} #{emp['lastName']&.downcase}",
29
+ "#{emp['preferredName']&.downcase} #{emp['lastName']&.downcase}",
30
+ "#{emp['firstName']&.downcase} #{emp['customPreferredLastName']&.downcase}",
31
+ "#{emp['preferredName']&.downcase} #{emp['customPreferredLastName']&.downcase}"
32
+ ].include?(name.downcase)
33
+ end
34
+ end
35
+ alias_method :search_team_member, :search_employee
36
+
37
+ def search_employee_by_field(field:, value:)
38
+ employees.find { |employee| employee[field] == value.to_s }
39
+ end
40
+ alias_method :search_team_member_by_field, :search_employee_by_field
41
+
42
+ def fetch_manager(team_member)
43
+ active_team_members.find { |tm| tm['id'] == team_member['supervisorEId'] }
44
+ end
45
+
46
+ def fetch_second_level_manager(team_member)
47
+ fetch_manager(fetch_manager(team_member))
48
+ end
49
+
50
+ def create_employee(employee_details_hash)
51
+ @client.employee.add(employee_details_hash)
52
+ end
53
+
54
+ def update_employee(employee_id, employee_details_hash)
55
+ @client.employee.update(employee_id, employee_details_hash)
56
+ end
57
+
58
+ def departments
59
+ meta_fields.detect { |res| res['name'] == 'Department' }.dig('options').each_with_object([]) { |option, array| array << option['name'] if option['archived'] == 'no' } || []
60
+ end
61
+
62
+ def divisions
63
+ meta_fields.detect { |res| res['name'] == 'Division' }.dig('options').each_with_object([]) { |option, array| array << option['name'] if option['archived'] == 'no' } || []
64
+ end
65
+
66
+ def fields
67
+ @fields ||= (Bamboozled::API::FieldCollection.all_names + @client.meta.fields.map { |f| f['alias'] }).compact.uniq
68
+ end
69
+
70
+ def employees
71
+ retries = 0
72
+
73
+ begin
74
+ @employees ||= @client.report.custom(fields, 'JSON').reject { |employee| employee['lastName'] == 'Test-Gitlab' }
75
+ rescue Net::ReadTimeout, Bamboozled::GatewayError
76
+ retry if (retries += 1) < MAX_RETRIES
77
+ end
78
+ end
79
+
80
+ def active_employees
81
+ employees.select { |employee| employee['status'] == 'Active' }
82
+ end
83
+ alias_method :active_team_members, :active_employees
84
+
85
+ def add_stock_options(employee_id, data)
86
+ @client.employee.add_table_row(employee_id, 'customEquity', data)
87
+ end
88
+
89
+ def stock_options_data(employee_id)
90
+ @client.employee.table_data(employee_id, 'customEquity')
91
+ end
92
+
93
+ def update_job_details(employee_id, data)
94
+ current_data = job_details(employee_id) # it should only be one row as we just created this user
95
+ row_id = current_data.first.dig('id')
96
+ @client.employee.update_table_row(employee_id, 'jobInfo', row_id, data)
97
+ end
98
+
99
+ def add_compensation_details(employee_id, data)
100
+ @client.employee.add_table_row(employee_id, 'compensation', data)
101
+ end
102
+
103
+ def employment_statuses(employee_id)
104
+ @client.employee.table_data(employee_id, 'employmentStatus')
105
+ end
106
+
107
+ def job_details(employee_id)
108
+ @client.employee.table_data(employee_id, 'jobInfo')
109
+ end
110
+
111
+ def resumes_folder_id(employee_id)
112
+ @resumes_folder_id ||= files(employee_id).dig('categories').find { |folder| folder['name'] == 'Resumes and Applications' }.dig('id')
113
+ end
114
+
115
+ def contract_folder_id(employee_id)
116
+ @contract_folder_id ||= files(employee_id).dig('categories').find { |folder| folder['name'] == 'Contracts & Changes' }.dig('id')
117
+ end
118
+
119
+ def add_file(employee_id, file_name, file, folder_id)
120
+ options = {
121
+ category: folder_id,
122
+ fileName: file_name,
123
+ share: 'yes',
124
+ file: file
125
+ }
126
+ @client.employee.add_file(employee_id, options)
127
+ end
128
+
129
+ def add_employment_status(employee_id, data)
130
+ @client.employee.add_table_row(employee_id, 'employmentStatus', data)
131
+ end
132
+
133
+ def add_currency_conversion(employee_id, data)
134
+ @client.employee.add_table_row(employee_id, 'customCurrencyConversion', data)
135
+ end
136
+
137
+ def add_on_target_earnings(employee_id, data)
138
+ @client.employee.add_table_row(employee_id, 'customOnTargetEarnings', data)
139
+ end
140
+
141
+ def add_signing_bonus(employee_id, data)
142
+ @client.employee.add_table_row(employee_id, 'customBonus', data)
143
+ end
144
+
145
+ def add_family_member(employee_id, data)
146
+ @client.employee.add_table_row(employee_id, 'customFamilyMember', data)
147
+ end
148
+
149
+ def add_additional_data(employee_id, data)
150
+ @client.employee.add_table_row(employee_id, 'customAdditionalInformation1', data)
151
+ end
152
+
153
+ def add_bonus(team_member_id, comment)
154
+ data = {
155
+ customBonusdate: Date.today.to_s,
156
+ customBonusamount: { value: 1_000, currency: 'USD' },
157
+ customBonustype: 'Discretionary Bonus',
158
+ customBonuscomments: comment
159
+ # customNominatedBy: 'TODO'
160
+ }
161
+ @client.employee.add_table_row(team_member_id, 'customBonus', data)
162
+ end
163
+
164
+ private
165
+
166
+ def meta_fields
167
+ @meta_fields ||= @client.meta.lists
168
+ end
169
+
170
+ def files(employee_id)
171
+ @files ||= @client.employee.files(employee_id)
172
+ end
173
+ end
174
+ end
175
+ end
@@ -45,7 +45,25 @@ module PeopleGroup
45
45
  epic || @client.create_epic(group_id, title, options)
46
46
  end
47
47
 
48
- def commit_change_to_new_merge_request_v2(project_id, branch_name, commit_message, description = '', files_to_delete = [], files_to_update = [])
48
+ def commit_new_files_to_new_merge_request(project_id, branch_name, new_files, commit_message, description = nil)
49
+ @client.create_branch(project_id, branch_name, 'master')
50
+ actions = []
51
+ new_files.each do |file|
52
+ actions << {
53
+ action: 'create',
54
+ file_path: file.sub('tmp', '/data/team_members/person'),
55
+ content: File.read(file)
56
+ }
57
+ end
58
+
59
+ @client.create_commit(project_id, branch_name, commit_message, actions)
60
+
61
+ options = { source_branch: branch_name, target_branch: 'master', remove_source_branch: true }
62
+ options[:description] = description if description
63
+ @client.create_merge_request(project_id, commit_message, options)
64
+ end
65
+
66
+ def commit_change_to_new_merge_request_v2(project_id, branch_name, commit_message, description = nil, files_to_delete = [], files_to_update = [])
49
67
  actions = []
50
68
 
51
69
  files_to_delete.each do |file|
@@ -71,7 +89,7 @@ module PeopleGroup
71
89
  @client.create_commit(project_id, branch_name, commit_message, actions)
72
90
 
73
91
  options = { source_branch: branch_name, target_branch: 'master', remove_source_branch: true }
74
- options[:description] = description if description.present?
92
+ options[:description] = description if description
75
93
  @client.create_merge_request(project_id, commit_message, options)
76
94
  end
77
95
 
@@ -0,0 +1,81 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'greenhouse_io'
4
+
5
+ module PeopleGroup
6
+ module Connectors
7
+ class Greenhouse
8
+ MAX_RETRIES = 3
9
+
10
+ def initialize(use_users_api_key: false)
11
+ api_key = use_users_api_key ? ENV['GREENHOUSE_API_KEY_USERS'] : ENV['GREENHOUSE_API_KEY']
12
+ @client = GreenhouseIo::Client.new(api_key)
13
+ end
14
+
15
+ def offer_for_application(application_id)
16
+ @client.offers_for_application(application_id)
17
+ end
18
+
19
+ def hired_candidates(updated_since)
20
+ page = 1
21
+ candidates = []
22
+ retries = 0
23
+
24
+ loop do
25
+ begin
26
+ results = @client.candidates(nil, updated_after: updated_since, page: page)
27
+ break if results.empty?
28
+
29
+ results.each do |candidate|
30
+ candidates << candidate if hired_non_active?(candidate)
31
+ end
32
+ rescue GreenhouseIo::Error
33
+ p [updated_since, page]
34
+ retry if (retries += 1) < MAX_RETRIES
35
+ end
36
+ page += 1
37
+ end
38
+
39
+ candidates
40
+ end
41
+
42
+ def candidate(candidate_id)
43
+ @client.candidates(candidate_id)
44
+ end
45
+
46
+ def add_sync_note_to_candidate(candidate_id)
47
+ note = {
48
+ user_id: ENV['GREENHOUSE_AUTHOR_ID'],
49
+ body: "This person was synced at #{Time.now} by the Employee Bot",
50
+ visibility: 'public'
51
+ }
52
+ @client.create_candidate_note(candidate_id, note, ENV['GREENHOUSE_AUTHOR_ID'])
53
+ end
54
+
55
+ def users
56
+ page = 1
57
+ users = []
58
+
59
+ loop do
60
+ results = @client.users(nil, page: page)
61
+ break if results.empty?
62
+
63
+ users += results
64
+ page += 1
65
+ end
66
+
67
+ users
68
+ end
69
+
70
+ private
71
+
72
+ def hired_non_active?(candidate)
73
+ # If the candidate has any application that is active, we don't sync.
74
+ return false if candidate['applications'].any? { |application| application['status'] == 'active' }
75
+
76
+ # Check if candidate is hired for just one of their applications
77
+ candidate['applications'].one? { |application| application['status'] == 'hired' }
78
+ end
79
+ end
80
+ end
81
+ end
@@ -12,20 +12,38 @@ module PeopleGroup
12
12
  @client = ::Slack::Web::Client.new
13
13
  end
14
14
 
15
- def send_message(channel, message)
16
- @client.chat_postMessage(channel: channel, text: message, as_user: true)
17
- end
18
-
19
15
  def find_user(email)
20
16
  @client.users_lookupByEmail(email: email)
21
17
  rescue ::Slack::Web::Api::Errors::SlackError
22
18
  nil
23
19
  end
24
20
 
21
+ def find_user_by_id(id)
22
+ @client.users_info(user: id)
23
+ rescue ::Slack::Web::Api::Errors::SlackError
24
+ nil
25
+ end
26
+
25
27
  def list_public_channels
26
28
  channels = @client.channels_list(exclude_archived: true).channels
27
29
  channels.reject(&:is_private)&.map(:name)
28
30
  end
31
+
32
+ def send_message(channel:, text:, as_user: true, attachments: [])
33
+ @client.chat_postMessage(channel: channel, text: text, as_user: true, attachments: attachments, unfurl_links: false)
34
+ end
35
+
36
+ def update_message(channel:, timestamp:, text:)
37
+ @client.chat_update(channel: channel, text: text, ts: timestamp, as_user: true, attachments: [])
38
+ end
39
+
40
+ def send_modal_message(trigger:, view:)
41
+ @client.views_open(trigger_id: trigger, view: view)
42
+ end
43
+
44
+ def publish_view(user_id:, trigger:, view:)
45
+ @client.views_publish(user_id: user_id, trigger_id: trigger, view: view)
46
+ end
29
47
  end
30
48
  end
31
49
  end
@@ -2,6 +2,6 @@
2
2
 
3
3
  module PeopleGroup
4
4
  module Connectors
5
- VERSION = '0.1.8'
5
+ VERSION = '0.1.13'
6
6
  end
7
7
  end
metadata CHANGED
@@ -1,14 +1,14 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: peoplegroup-connectors
3
3
  version: !ruby/object:Gem::Version
4
- version: 0.1.8
4
+ version: 0.1.13
5
5
  platform: ruby
6
6
  authors:
7
7
  - lien van den steen
8
8
  autorequire:
9
9
  bindir: bin
10
10
  cert_chain: []
11
- date: 2021-02-02 00:00:00.000000000 Z
11
+ date: 2021-02-08 00:00:00.000000000 Z
12
12
  dependencies:
13
13
  - !ruby/object:Gem::Dependency
14
14
  name: gitlab
@@ -52,6 +52,34 @@ dependencies:
52
52
  - - "~>"
53
53
  - !ruby/object:Gem::Version
54
54
  version: '0.1'
55
+ - !ruby/object:Gem::Dependency
56
+ name: bamboozled-gitlab
57
+ requirement: !ruby/object:Gem::Requirement
58
+ requirements:
59
+ - - "~>"
60
+ - !ruby/object:Gem::Version
61
+ version: 0.2.9
62
+ type: :runtime
63
+ prerelease: false
64
+ version_requirements: !ruby/object:Gem::Requirement
65
+ requirements:
66
+ - - "~>"
67
+ - !ruby/object:Gem::Version
68
+ version: 0.2.9
69
+ - !ruby/object:Gem::Dependency
70
+ name: greenhouse_io-gitlab
71
+ requirement: !ruby/object:Gem::Requirement
72
+ requirements:
73
+ - - "~>"
74
+ - !ruby/object:Gem::Version
75
+ version: '2.5'
76
+ type: :runtime
77
+ prerelease: false
78
+ version_requirements: !ruby/object:Gem::Requirement
79
+ requirements:
80
+ - - "~>"
81
+ - !ruby/object:Gem::Version
82
+ version: '2.5'
55
83
  - !ruby/object:Gem::Dependency
56
84
  name: rspec
57
85
  requirement: !ruby/object:Gem::Requirement
@@ -72,14 +100,28 @@ dependencies:
72
100
  requirements:
73
101
  - - "~>"
74
102
  - !ruby/object:Gem::Version
75
- version: '0.85'
103
+ version: 0.91.1
104
+ type: :development
105
+ prerelease: false
106
+ version_requirements: !ruby/object:Gem::Requirement
107
+ requirements:
108
+ - - "~>"
109
+ - !ruby/object:Gem::Version
110
+ version: 0.91.1
111
+ - !ruby/object:Gem::Dependency
112
+ name: gitlab-styles
113
+ requirement: !ruby/object:Gem::Requirement
114
+ requirements:
115
+ - - "~>"
116
+ - !ruby/object:Gem::Version
117
+ version: '6.0'
76
118
  type: :development
77
119
  prerelease: false
78
120
  version_requirements: !ruby/object:Gem::Requirement
79
121
  requirements:
80
122
  - - "~>"
81
123
  - !ruby/object:Gem::Version
82
- version: '0.85'
124
+ version: '6.0'
83
125
  - !ruby/object:Gem::Dependency
84
126
  name: rubocop-packaging
85
127
  requirement: !ruby/object:Gem::Requirement
@@ -118,7 +160,9 @@ files:
118
160
  - LICENSE.txt
119
161
  - README.md
120
162
  - lib/peoplegroup/connectors.rb
163
+ - lib/peoplegroup/connectors/bamboo.rb
121
164
  - lib/peoplegroup/connectors/gitlab.rb
165
+ - lib/peoplegroup/connectors/greenhouse.rb
122
166
  - lib/peoplegroup/connectors/pto_roots.rb
123
167
  - lib/peoplegroup/connectors/slack.rb
124
168
  - lib/peoplegroup/connectors/version.rb