introspective_admin 0.1.0 → 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.
Files changed (121) hide show
  1. checksums.yaml +4 -4
  2. data/.DS_Store +0 -0
  3. data/.gitignore +1 -0
  4. data/.rubocop.yml +75 -0
  5. data/.ruby-version +1 -1
  6. data/.travis.yml +5 -14
  7. data/CHANGELOG.md +37 -28
  8. data/Gemfile +24 -6
  9. data/Gemfile.lock +402 -264
  10. data/README.md +34 -8
  11. data/Rakefile +3 -5
  12. data/introspective_admin.gemspec +29 -44
  13. data/lib/introspective_admin/base.rb +217 -200
  14. data/lib/introspective_admin/version.rb +5 -3
  15. data/lib/introspective_admin.rb +2 -0
  16. data/lib/tasks/introspective_admin_tasks.rake +2 -0
  17. data/spec/admin/company_admin_spec.rb +73 -72
  18. data/spec/admin/job_admin_spec.rb +63 -61
  19. data/spec/admin/location_admin_spec.rb +70 -66
  20. data/spec/admin/location_beacon_admin_spec.rb +75 -73
  21. data/spec/admin/project__admin_spec.rb +73 -71
  22. data/spec/admin/user_admin_spec.rb +65 -64
  23. data/spec/dummy/Gemfile +17 -0
  24. data/spec/dummy/README.rdoc +28 -28
  25. data/spec/dummy/Rakefile +8 -6
  26. data/spec/dummy/app/admin/admin_users.rb +29 -0
  27. data/spec/dummy/app/admin/company_admin.rb +5 -4
  28. data/spec/dummy/app/admin/dashboard.rb +34 -0
  29. data/spec/dummy/app/admin/job_admin.rb +5 -4
  30. data/spec/dummy/app/admin/location_admin.rb +5 -4
  31. data/spec/dummy/app/admin/location_beacon_admin.rb +8 -6
  32. data/spec/dummy/app/admin/project_admin.rb +5 -6
  33. data/spec/dummy/app/admin/role_admin.rb +5 -5
  34. data/spec/dummy/app/admin/user_admin.rb +13 -13
  35. data/spec/dummy/app/assets/config/manifest.js +3 -0
  36. data/spec/dummy/app/assets/javascripts/active_admin.js +1 -0
  37. data/spec/dummy/app/assets/javascripts/application.js +13 -13
  38. data/spec/dummy/app/assets/stylesheets/active_admin.scss +17 -0
  39. data/spec/dummy/app/assets/stylesheets/application.css +15 -15
  40. data/spec/dummy/app/controllers/application_controller.rb +10 -8
  41. data/spec/dummy/app/helpers/application_helper.rb +4 -3
  42. data/spec/dummy/app/models/abstract_adapter.rb +20 -12
  43. data/spec/dummy/app/models/admin_user.rb +12 -6
  44. data/spec/dummy/app/models/company.rb +13 -12
  45. data/spec/dummy/app/models/job.rb +10 -10
  46. data/spec/dummy/app/models/locatable.rb +8 -6
  47. data/spec/dummy/app/models/location.rb +27 -26
  48. data/spec/dummy/app/models/location_beacon.rb +19 -19
  49. data/spec/dummy/app/models/location_gps.rb +11 -11
  50. data/spec/dummy/app/models/project.rb +20 -20
  51. data/spec/dummy/app/models/project_job.rb +8 -7
  52. data/spec/dummy/app/models/role.rb +26 -25
  53. data/spec/dummy/app/models/team.rb +10 -9
  54. data/spec/dummy/app/models/team_user.rb +14 -13
  55. data/spec/dummy/app/models/user.rb +72 -68
  56. data/spec/dummy/app/models/user_location.rb +28 -28
  57. data/spec/dummy/app/models/user_project_job.rb +17 -16
  58. data/spec/dummy/app/views/layouts/application.html.erb +13 -13
  59. data/spec/dummy/bin/bundle +5 -3
  60. data/spec/dummy/bin/rails +6 -4
  61. data/spec/dummy/bin/rake +6 -4
  62. data/spec/dummy/bin/setup +31 -29
  63. data/spec/dummy/config/application.rb +35 -34
  64. data/spec/dummy/config/boot.rb +7 -5
  65. data/spec/dummy/config/database.yml +22 -22
  66. data/spec/dummy/config/environment.rb +12 -11
  67. data/spec/dummy/config/environments/development.rb +47 -45
  68. data/spec/dummy/config/environments/production.rb +83 -82
  69. data/spec/dummy/config/environments/test.rb +50 -50
  70. data/spec/dummy/config/initializers/active_admin.rb +8 -7
  71. data/spec/dummy/config/initializers/assets.rb +15 -13
  72. data/spec/dummy/config/initializers/backtrace_silencers.rb +9 -7
  73. data/spec/dummy/config/initializers/cookies_serializer.rb +5 -3
  74. data/spec/dummy/config/initializers/devise.rb +265 -263
  75. data/spec/dummy/config/initializers/filter_parameter_logging.rb +6 -4
  76. data/spec/dummy/config/initializers/inflections.rb +18 -16
  77. data/spec/dummy/config/initializers/mime_types.rb +6 -4
  78. data/spec/dummy/config/initializers/session_store.rb +5 -3
  79. data/spec/dummy/config/initializers/wrap_parameters.rb +16 -14
  80. data/spec/dummy/config/initializers/zeitwerk.rb +10 -0
  81. data/spec/dummy/config/locales/devise.en.yml +60 -60
  82. data/spec/dummy/config/locales/en.yml +23 -23
  83. data/spec/dummy/config/routes.rb +12 -9
  84. data/spec/dummy/config/secrets.yml +20 -20
  85. data/spec/dummy/config.ru +6 -4
  86. data/spec/dummy/db/development.sqlite3 +0 -0
  87. data/spec/dummy/db/migrate/20141002205024_devise_create_users.rb +43 -42
  88. data/spec/dummy/db/migrate/20141002211055_devise_create_admin_users.rb +49 -48
  89. data/spec/dummy/db/migrate/20141002211057_create_active_admin_comments.rb +21 -19
  90. data/spec/dummy/db/migrate/20141002220722_add_lockable_to_users.rb +10 -8
  91. data/spec/dummy/db/migrate/20150406213646_create_companies.rb +13 -11
  92. data/spec/dummy/db/migrate/20150414213154_add_user_authentication_token.rb +13 -11
  93. data/spec/dummy/db/migrate/20150415222005_create_roles.rb +13 -12
  94. data/spec/dummy/db/migrate/20150505181635_create_chats.rb +11 -9
  95. data/spec/dummy/db/migrate/20150505181636_create_chat_users.rb +13 -11
  96. data/spec/dummy/db/migrate/20150505181640_create_chat_messages.rb +13 -11
  97. data/spec/dummy/db/migrate/20150507191529_create_chat_message_users.rb +13 -11
  98. data/spec/dummy/db/migrate/20150601200526_create_locations.rb +15 -13
  99. data/spec/dummy/db/migrate/20150601200533_create_locatables.rb +12 -10
  100. data/spec/dummy/db/migrate/20150601212924_create_location_beacons.rb +18 -16
  101. data/spec/dummy/db/migrate/20150601213542_create_location_gps.rb +14 -12
  102. data/spec/dummy/db/migrate/20150609201823_create_user_locations.rb +16 -14
  103. data/spec/dummy/db/migrate/20150617232519_create_projects.rb +12 -10
  104. data/spec/dummy/db/migrate/20150617232521_create_jobs.rb +11 -9
  105. data/spec/dummy/db/migrate/20150617232522_create_project_jobs.rb +13 -11
  106. data/spec/dummy/db/migrate/20150623170133_create_user_project_jobs.rb +14 -12
  107. data/spec/dummy/db/migrate/20150701234929_create_teams.rb +13 -11
  108. data/spec/dummy/db/migrate/20150701234930_create_team_users.rb +13 -11
  109. data/spec/dummy/db/migrate/20150727214950_add_confirmable_to_devise.rb +13 -11
  110. data/spec/dummy/db/migrate/20150820190524_add_user_names.rb +8 -6
  111. data/spec/dummy/db/migrate/20150909225019_add_password_to_project.rb +7 -5
  112. data/spec/dummy/db/migrate/20220806003731_add_devise_to_admin_users.rb +50 -0
  113. data/spec/dummy/db/schema.rb +263 -264
  114. data/spec/dummy/public/404.html +67 -67
  115. data/spec/dummy/public/422.html +67 -67
  116. data/spec/dummy/public/500.html +66 -66
  117. data/spec/rails_helper.rb +33 -27
  118. data/spec/support/blueprints.rb +50 -49
  119. data/spec/support/location_helper.rb +28 -29
  120. metadata +21 -252
  121. data/Gemfile.lock.rails4 +0 -256
@@ -1,12 +1,20 @@
1
- class AbstractAdapter < ActiveRecord::Base
2
- self.abstract_class = true
3
-
4
- class << self
5
- def human_attribute_name(attr, options = {})
6
- # The default formatting of validation errors sucks, this helps a little syntatically:
7
- super.titleize+":"
8
- end
9
- end
10
-
11
- end
12
-
1
+ # frozen_string_literal: true
2
+
3
+ class AbstractAdapter < ActiveRecord::Base
4
+ self.abstract_class = true
5
+
6
+ class << self
7
+ def human_attribute_name(attr, options = {})
8
+ # The default formatting of validation errors sucks, this helps a little syntatically:
9
+ "#{super.titleize}:"
10
+ end
11
+
12
+ def ransackable_attributes(_auth_object = nil)
13
+ @ransackable_attributes ||= column_names + _ransackers.keys
14
+ end
15
+
16
+ def ransackable_associations(_auth_object = nil)
17
+ @ransackable_associations ||= reflect_on_all_associations.map { |a| a.name.to_s } + _ransackers.keys
18
+ end
19
+ end
20
+ end
@@ -1,6 +1,12 @@
1
- class AdminUser < ActiveRecord::Base
2
- # Include default devise modules. Others available are:
3
- # :confirmable, :lockable, :timeoutable and :omniauthable
4
- devise :database_authenticatable,
5
- :recoverable, :rememberable, :trackable, :validatable
6
- end
1
+ # frozen_string_literal: true
2
+
3
+ class AdminUser < ActiveRecord::Base
4
+ # Include default devise modules. Others available are:
5
+ # :confirmable, :lockable, :timeoutable, :trackable and :omniauthable
6
+ devise :database_authenticatable,
7
+ :recoverable, :rememberable, :validatable
8
+ # Include default devise modules. Others available are:
9
+ # :confirmable, :lockable, :timeoutable and :omniauthable
10
+ devise :database_authenticatable,
11
+ :recoverable, :rememberable, :trackable, :validatable
12
+ end
@@ -1,12 +1,13 @@
1
- class Company < AbstractAdapter
2
- has_many :roles, as: :ownable
3
- has_many :admins, through: :roles, source: :user
4
- accepts_nested_attributes_for :roles, allow_destroy: true
5
-
6
- has_many :beacons, class_name: 'LocationBeacon', dependent: :destroy
7
- has_many :locatables
8
- has_many :locations, through: :locatables, source: :locatable, source_type: 'Company'
9
-
10
- has_many :projects, foreign_key: :owner_id, dependent: :destroy, inverse_of: :owner
11
-
12
- end
1
+ # frozen_string_literal: true
2
+
3
+ class Company < AbstractAdapter
4
+ has_many :roles, as: :ownable
5
+ has_many :admins, through: :roles, source: :user
6
+ accepts_nested_attributes_for :roles, allow_destroy: true
7
+
8
+ has_many :beacons, class_name: 'LocationBeacon', dependent: :destroy
9
+ has_many :locatables
10
+ has_many :locations, through: :locatables, source: :locatable, source_type: 'Company'
11
+
12
+ has_many :projects, foreign_key: :owner_id, dependent: :destroy, inverse_of: :owner
13
+ end
@@ -1,10 +1,10 @@
1
- class Job < AbstractAdapter
2
-
3
- has_many :project_jobs, dependent: :destroy
4
- accepts_nested_attributes_for :project_jobs, allow_destroy: true
5
-
6
- has_many :user_project_jobs, dependent: :destroy
7
- has_many :users, through: :user_project_jobs
8
- has_many :projects, through: :user_project_jobs
9
-
10
- end
1
+ # frozen_string_literal: true
2
+
3
+ class Job < AbstractAdapter
4
+ has_many :project_jobs, dependent: :destroy
5
+ accepts_nested_attributes_for :project_jobs, allow_destroy: true
6
+
7
+ has_many :user_project_jobs, dependent: :destroy
8
+ has_many :users, through: :user_project_jobs
9
+ has_many :projects, through: :user_project_jobs
10
+ end
@@ -1,6 +1,8 @@
1
- class Locatable < AbstractAdapter
2
- belongs_to :location
3
- belongs_to :locatable, polymorphic: true
4
-
5
- validates_inclusion_of :locatable_type, in: %w(Company)
6
- end
1
+ # frozen_string_literal: true
2
+
3
+ class Locatable < AbstractAdapter
4
+ belongs_to :location
5
+ belongs_to :locatable, polymorphic: true
6
+
7
+ validates_inclusion_of :locatable_type, in: %w[Company]
8
+ end
@@ -1,26 +1,27 @@
1
- class Location < AbstractAdapter
2
- has_many :locatables, dependent: :destroy
3
- has_many :companies, through: :locatables, source: :locatable, source_type: 'Company'
4
-
5
- has_many :beacons, class_name: 'LocationBeacon', dependent: :destroy
6
- has_one :gps, class_name: 'LocationGps', dependent: :destroy
7
- delegate :lat,:lng,:alt, to: :gps
8
-
9
- belongs_to :parent_location, foreign_key: :parent_location_id, class_name: 'Location', inverse_of: :child_locations
10
- has_many :child_locations, foreign_key: :parent_location_id, class_name: 'Location', dependent: :destroy, inverse_of: :parent_location
11
-
12
- has_many :user_locations, dependent: :destroy
13
-
14
- # isn't this list going to be kinda long? are there any reasonable constraints to put
15
- # on this random bit of metadata?
16
- validates_inclusion_of :kind, in: %w(airport terminal gate plane)
17
-
18
- accepts_nested_attributes_for :child_locations, allow_destroy: true
19
- accepts_nested_attributes_for :gps, allow_destroy: true
20
- accepts_nested_attributes_for :beacons, allow_destroy: true
21
-
22
- def coords
23
- [gps.lat, gps.lng, gps.alt]
24
- end
25
-
26
- end
1
+ # frozen_string_literal: true
2
+
3
+ class Location < AbstractAdapter
4
+ has_many :locatables, dependent: :destroy
5
+ has_many :companies, through: :locatables, source: :locatable, source_type: 'Company'
6
+
7
+ has_many :beacons, class_name: 'LocationBeacon', dependent: :destroy
8
+ has_one :gps, class_name: 'LocationGps', dependent: :destroy
9
+ delegate :lat, :lng, :alt, to: :gps
10
+
11
+ belongs_to :parent_location, foreign_key: :parent_location_id, class_name: 'Location', inverse_of: :child_locations
12
+ has_many :child_locations, foreign_key: :parent_location_id, class_name: 'Location', dependent: :destroy, inverse_of: :parent_location
13
+
14
+ has_many :user_locations, dependent: :destroy
15
+
16
+ # isn't this list going to be kinda long? are there any reasonable constraints to put
17
+ # on this random bit of metadata?
18
+ validates_inclusion_of :kind, in: %w[airport terminal gate plane]
19
+
20
+ accepts_nested_attributes_for :child_locations, allow_destroy: true
21
+ accepts_nested_attributes_for :gps, allow_destroy: true
22
+ accepts_nested_attributes_for :beacons, allow_destroy: true
23
+
24
+ def coords
25
+ [gps.lat, gps.lng, gps.alt]
26
+ end
27
+ end
@@ -1,19 +1,19 @@
1
- class LocationBeacon < AbstractAdapter
2
- belongs_to :location
3
- has_many :gps, through: :location
4
- belongs_to :company
5
-
6
- # B9407F30-F5F8-466E-AFF9-25556B57FE6D
7
- validates_format_of :uuid, with: /[0-9a-fA-F]{32}/ # 32 digit hexadecimal UUID
8
- validates_format_of :mac_address, with: /[0-9a-fA-F]{12}/ # 16 digit hexadecimal bluetooth MAC address
9
-
10
- before_validation :massage_ids
11
- def massage_ids
12
- self.uuid = (uuid||'').gsub(/[^0-9a-fA-F]+/,'').upcase
13
- self.mac_address = (mac_address||'').gsub(/[^0-9a-fA-F]+/,'').upcase
14
- end
15
-
16
- scope :low_battery, -> { where("last_known_battery_level < 20") }
17
-
18
-
19
- end
1
+ # frozen_string_literal: true
2
+
3
+ class LocationBeacon < AbstractAdapter
4
+ belongs_to :location
5
+ has_many :gps, through: :location
6
+ belongs_to :company
7
+
8
+ # B9407F30-F5F8-466E-AFF9-25556B57FE6D
9
+ validates_format_of :uuid, with: /[0-9a-fA-F]{32}/ # 32 digit hexadecimal UUID
10
+ validates_format_of :mac_address, with: /[0-9a-fA-F]{12}/ # 16 digit hexadecimal bluetooth MAC address
11
+
12
+ before_validation :massage_ids
13
+ def massage_ids
14
+ self.uuid = (uuid || '').gsub(/[^0-9a-fA-F]+/, '').upcase
15
+ self.mac_address = (mac_address || '').gsub(/[^0-9a-fA-F]+/, '').upcase
16
+ end
17
+
18
+ scope :low_battery, -> { where('last_known_battery_level < 20') }
19
+ end
@@ -1,11 +1,11 @@
1
- class LocationGps < AbstractAdapter
2
- belongs_to :location
3
- has_many :beacons, through: :location
4
-
5
- # lat and lng in degrees altitude in meters
6
- validates_numericality_of :lat, greater_than_or_equal_to: -90.0, less_than_or_equal_to: 90.0
7
- validates_numericality_of :lng, greater_than_or_equal_to: -180.0, less_than_or_equal_to: 180.0
8
- validates_numericality_of :alt
9
-
10
-
11
- end
1
+ # frozen_string_literal: true
2
+
3
+ class LocationGps < AbstractAdapter
4
+ belongs_to :location
5
+ has_many :beacons, through: :location
6
+
7
+ # lat and lng in degrees altitude in meters
8
+ validates_numericality_of :lat, greater_than_or_equal_to: -90.0, less_than_or_equal_to: 90.0
9
+ validates_numericality_of :lng, greater_than_or_equal_to: -180.0, less_than_or_equal_to: 180.0
10
+ validates_numericality_of :alt
11
+ end
@@ -1,20 +1,20 @@
1
- class Project < AbstractAdapter
2
- belongs_to :owner, foreign_key: :owner_id, class_name: 'Company'
3
-
4
- has_many :roles, as: :ownable
5
- has_many :admins, through: :roles, source: :user
6
- accepts_nested_attributes_for :roles, allow_destroy: true
7
-
8
- has_many :project_jobs, dependent: :destroy, inverse_of: :project
9
- has_many :jobs, through: :project_jobs
10
- accepts_nested_attributes_for :project_jobs, allow_destroy: true
11
-
12
- has_many :user_project_jobs, dependent: :destroy, inverse_of: :project
13
- has_many :users, through: :user_project_jobs, inverse_of: :projects
14
- accepts_nested_attributes_for :user_project_jobs, allow_destroy: true
15
-
16
- has_many :teams, dependent: :destroy, inverse_of: :project
17
- accepts_nested_attributes_for :teams, allow_destroy: true
18
-
19
-
20
- end
1
+ # frozen_string_literal: true
2
+
3
+ class Project < AbstractAdapter
4
+ belongs_to :owner, foreign_key: :owner_id, class_name: 'Company'
5
+
6
+ has_many :roles, as: :ownable
7
+ has_many :admins, through: :roles, source: :user
8
+ accepts_nested_attributes_for :roles, allow_destroy: true
9
+
10
+ has_many :project_jobs, dependent: :destroy, inverse_of: :project
11
+ has_many :jobs, through: :project_jobs
12
+ accepts_nested_attributes_for :project_jobs, allow_destroy: true
13
+
14
+ has_many :user_project_jobs, dependent: :destroy, inverse_of: :project
15
+ has_many :users, through: :user_project_jobs, inverse_of: :projects
16
+ accepts_nested_attributes_for :user_project_jobs, allow_destroy: true
17
+
18
+ has_many :teams, dependent: :destroy, inverse_of: :project
19
+ accepts_nested_attributes_for :teams, allow_destroy: true
20
+ end
@@ -1,7 +1,8 @@
1
- class ProjectJob < AbstractAdapter
2
- belongs_to :project
3
- belongs_to :job
4
-
5
- delegate :title, to: :job
6
-
7
- end
1
+ # frozen_string_literal: true
2
+
3
+ class ProjectJob < AbstractAdapter
4
+ belongs_to :project
5
+ belongs_to :job
6
+
7
+ delegate :title, to: :job
8
+ end
@@ -1,25 +1,26 @@
1
- class Role < AbstractAdapter
2
- belongs_to :user
3
- belongs_to :ownable, polymorphic: true
4
-
5
- validates_uniqueness_of :user_id, scope: [:ownable_type,:ownable_id], unless: Proc.new {|u| u.user_id.nil? }, message: "user has already been assigned that role"
6
- validates_inclusion_of :ownable_type, in: ['Company', 'Project']
7
-
8
- delegate :email, to: :user, allow_nil: true
9
- def attributes
10
- scuper.merge(email: email)
11
- end
12
-
13
- def self.ownable_assign_options(model=nil)
14
- (Company.all + Project.all).map { |i| [ "#{i.class}: #{i.name}", "#{i.class}-#{i.id}"] }
15
- end
16
-
17
- def ownable_assign
18
- ownable.present? ? "#{ownable_type}-#{ownable_id}" : nil
19
- end
20
-
21
- def ownable_assign=(value)
22
- self.ownable_type,self.ownable_id = value.split('-')
23
- end
24
-
25
- end
1
+ # frozen_string_literal: true
2
+
3
+ class Role < AbstractAdapter
4
+ belongs_to :user
5
+ belongs_to :ownable, polymorphic: true
6
+
7
+ validates_uniqueness_of :user_id, scope: %i[ownable_type ownable_id], unless: proc { |u| u.user_id.nil? }, message: 'user has already been assigned that role'
8
+ validates_inclusion_of :ownable_type, in: %w[Company Project]
9
+
10
+ delegate :email, to: :user, allow_nil: true
11
+ def attributes
12
+ scuper.merge(email: email)
13
+ end
14
+
15
+ def self.ownable_assign_options(_model = nil)
16
+ (Company.all + Project.all).map { |i| ["#{i.class}: #{i.name}", "#{i.class}-#{i.id}"] }
17
+ end
18
+
19
+ def ownable_assign
20
+ ownable.present? ? "#{ownable_type}-#{ownable_id}" : nil
21
+ end
22
+
23
+ def ownable_assign=(value)
24
+ self.ownable_type, self.ownable_id = value.split('-')
25
+ end
26
+ end
@@ -1,9 +1,10 @@
1
- class Team < AbstractAdapter
2
- belongs_to :project
3
- belongs_to :creator, class_name: 'User'
4
-
5
- has_many :team_users, inverse_of: :team
6
- has_many :users, through: :team_users
7
- accepts_nested_attributes_for :team_users, allow_destroy: true
8
-
9
- end
1
+ # frozen_string_literal: true
2
+
3
+ class Team < AbstractAdapter
4
+ belongs_to :project
5
+ belongs_to :creator, class_name: 'User'
6
+
7
+ has_many :team_users, inverse_of: :team
8
+ has_many :users, through: :team_users
9
+ accepts_nested_attributes_for :team_users, allow_destroy: true
10
+ end
@@ -1,13 +1,14 @@
1
- class TeamUser < AbstractAdapter
2
- belongs_to :user
3
- belongs_to :team
4
-
5
- validate :user_on_project
6
-
7
- def user_on_project
8
- unless user && team && user.projects.include?(team.project)
9
- errors.add(:user, "#{user.try(:name)} is not on the #{team.try(:project).try(:name)} project")
10
- end
11
- end
12
-
13
- end
1
+ # frozen_string_literal: true
2
+
3
+ class TeamUser < AbstractAdapter
4
+ belongs_to :user
5
+ belongs_to :team
6
+
7
+ validate :user_on_project
8
+
9
+ def user_on_project
10
+ return if user && team && user.projects.include?(team.project)
11
+
12
+ errors.add(:user, "#{user.try(:name)} is not on the #{team.try(:project).try(:name)} project")
13
+ end
14
+ end
@@ -1,68 +1,72 @@
1
- class User < AbstractAdapter
2
- # Include default devise modules. Others available are:
3
- # :confirmable, :lockable, :timeoutable and :omniauthable
4
- devise :database_authenticatable, :registerable, :confirmable,
5
- :recoverable, :rememberable, :trackable, :validatable, :lockable
6
-
7
- scope :active, -> { where(:locked_at => nil) }
8
- scope :inactive, -> { where('locked_at is not null') }
9
-
10
- has_many :user_locations, dependent: :destroy
11
-
12
- has_many :user_project_jobs, dependent: :destroy, inverse_of: :user
13
- has_many :jobs, through: :user_project_jobs, inverse_of: :users
14
- has_many :projects, through: :user_project_jobs, inverse_of: :users
15
- accepts_nested_attributes_for :user_project_jobs
16
-
17
- before_validation :set_default_password_from_project, on: :create
18
-
19
- has_many :team_users
20
- has_many :teams, through: :team_users
21
-
22
- has_many :roles, dependent: :destroy, inverse_of: :user
23
- accepts_nested_attributes_for :roles, allow_destroy: true
24
- has_many :admin_companies, through: :roles, source: :ownable, source_type: Company
25
- has_many :admin_projects, through: :roles, source: :ownable, source_type: Project
26
-
27
- def all_admin_projects # aggregate companies' projects with project admin roles
28
- (admin_companies.map(&:projects)+admin_projects).flatten
29
- end
30
-
31
- def admin?(record)
32
- roles.detect{|r| r.ownable == record }.present?
33
- end
34
-
35
- def company_admin? # an admin of any company
36
- roles.detect{|r| r.ownable_type == 'Company' }.present?
37
- end
38
-
39
- def project_admin? # an admin of any project
40
- company_admin? || roles.detect{|r| r.ownable_type == 'Project' }.present?
41
- end
42
-
43
-
44
- def set_default_password_from_project
45
- self.password = user_project_jobs.first.try(:project).try(:default_password) if password.blank?
46
- end
47
-
48
- def name
49
- [first_name,last_name].delete_if(&:blank?).join(' ')
50
- end
51
-
52
- def avatar_url(size='medium')
53
- avatar.try(:file).try(:url,size)
54
- end
55
-
56
- def self.attribute_param_types
57
- { "skip_confirmation_email" => Virtus::Attribute::Boolean }
58
- end
59
-
60
- def skip_confirmation_email=(s)
61
- return unless s.to_s == "true"
62
- # skip_confirmation! does not work with update_attributes, a work-around:
63
- self.update_column(:email, email) && self.reload if self.valid? && self.id
64
- # devise: confirm the user without requiring a confirmation email
65
- self.skip_confirmation!
66
- end
67
-
68
- end
1
+ # frozen_string_literal: true
2
+
3
+ class User < AbstractAdapter
4
+ # Include default devise modules. Others available are:
5
+ # :confirmable, :lockable, :timeoutable and :omniauthable
6
+ devise :database_authenticatable, :registerable, :confirmable,
7
+ :recoverable, :rememberable, :trackable, :validatable, :lockable
8
+
9
+ scope :active, -> { where(locked_at: nil) }
10
+ scope :inactive, -> { where('locked_at is not null') }
11
+
12
+ has_many :user_locations, dependent: :destroy
13
+
14
+ has_many :user_project_jobs, dependent: :destroy, inverse_of: :user
15
+ has_many :jobs, through: :user_project_jobs, inverse_of: :users
16
+ has_many :projects, through: :user_project_jobs, inverse_of: :users
17
+ accepts_nested_attributes_for :user_project_jobs
18
+
19
+ before_validation :set_default_password_from_project, on: :create
20
+
21
+ has_many :team_users
22
+ has_many :teams, through: :team_users
23
+
24
+ has_many :roles, dependent: :destroy, inverse_of: :user
25
+ accepts_nested_attributes_for :roles, allow_destroy: true
26
+ has_many :admin_companies, through: :roles, source: :ownable, source_type: 'Company'
27
+ has_many :admin_projects, through: :roles, source: :ownable, source_type: 'Project'
28
+
29
+ # aggregate companies' projects with project admin roles
30
+ def all_admin_projects
31
+ (admin_companies.map(&:projects) + admin_projects).flatten
32
+ end
33
+
34
+ def admin?(record)
35
+ roles.detect { |r| r.ownable == record }.present?
36
+ end
37
+
38
+ # an admin of any company
39
+ def company_admin?
40
+ roles.detect { |r| r.ownable_type == 'Company' }.present?
41
+ end
42
+
43
+ # an admin of any project
44
+ def project_admin?
45
+ company_admin? || roles.detect { |r| r.ownable_type == 'Project' }.present?
46
+ end
47
+
48
+ def set_default_password_from_project
49
+ self.password = user_project_jobs.first.try(:project).try(:default_password) if password.blank?
50
+ end
51
+
52
+ def name
53
+ [first_name, last_name].delete_if(&:blank?).join(' ')
54
+ end
55
+
56
+ def avatar_url(size = 'medium')
57
+ avatar.try(:file).try(:url, size)
58
+ end
59
+
60
+ def self.attribute_param_types
61
+ { 'skip_confirmation_email' => Virtus::Attribute::Boolean }
62
+ end
63
+
64
+ def skip_confirmation_email=(skip)
65
+ return unless skip.to_s == 'true'
66
+
67
+ # skip_confirmation! does not work with update_attributes, a work-around:
68
+ update_column(:email, email) && reload if valid? && id
69
+ # devise: confirm the user without requiring a confirmation email
70
+ skip_confirmation!
71
+ end
72
+ end
@@ -1,28 +1,28 @@
1
- class UserLocation < AbstractAdapter
2
- belongs_to :user
3
- belongs_to :location
4
- belongs_to :detectable, polymorphic: true
5
-
6
- validates_inclusion_of :detectable_type, in: %w(LocationBeacon LocationGps)
7
-
8
- default_scope { includes(:detectable).order("created_at desc") }
9
-
10
- def coords=(c) # convenience method to set coordinates by an array of [lat,lng,alt]
11
- self.lat = c[0]
12
- self.lng = c[1]
13
- self.alt = c[2]
14
- end
15
-
16
- def beacon
17
- detectable.is_a?(LocationBeacon) ? detectable : {}
18
- end
19
-
20
- def distance
21
- if location.gps && lat && lng
22
- location.gps.distance_from(lat,lng)
23
- else
24
- nil
25
- end
26
- end
27
-
28
- end
1
+ # frozen_string_literal: true
2
+
3
+ class UserLocation < AbstractAdapter
4
+ belongs_to :user
5
+ belongs_to :location
6
+ belongs_to :detectable, polymorphic: true
7
+
8
+ validates_inclusion_of :detectable_type, in: %w[LocationBeacon LocationGps]
9
+
10
+ default_scope { includes(:detectable).order('created_at desc') }
11
+
12
+ # convenience method to set coordinates by an array of [lat,lng,alt]
13
+ def coords=(coords)
14
+ self.lat = coords[0]
15
+ self.lng = coords[1]
16
+ self.alt = coords[2]
17
+ end
18
+
19
+ def beacon
20
+ detectable.is_a?(LocationBeacon) ? detectable : {}
21
+ end
22
+
23
+ def distance
24
+ return unless location.gps && lat && lng
25
+
26
+ location.gps.distance_from(lat, lng)
27
+ end
28
+ end
@@ -1,16 +1,17 @@
1
- class UserProjectJob < AbstractAdapter
2
- belongs_to :user, inverse_of: :user_project_jobs
3
- belongs_to :project, inverse_of: :user_project_jobs
4
- belongs_to :job, inverse_of: :user_project_jobs
5
-
6
- validates_inclusion_of :job, in: proc {|r| r.project.try(:jobs) || [] }
7
-
8
- delegate :email, :avatar_url, to: :user
9
- delegate :title, to: :job
10
- delegate :name, to: :project
11
-
12
- def self.options_for_job(project=nil)
13
- project.jobs
14
- end
15
-
16
- end
1
+ # frozen_string_literal: true
2
+
3
+ class UserProjectJob < AbstractAdapter
4
+ belongs_to :user, inverse_of: :user_project_jobs
5
+ belongs_to :project, inverse_of: :user_project_jobs
6
+ belongs_to :job, inverse_of: :user_project_jobs
7
+
8
+ validates_inclusion_of :job, in: proc { |r| r.project.try(:jobs) || [] }
9
+
10
+ delegate :email, :avatar_url, to: :user
11
+ delegate :title, to: :job
12
+ delegate :name, to: :project
13
+
14
+ def self.options_for_job(project = nil)
15
+ project.jobs
16
+ end
17
+ end