Merge branch 'dev'
This commit is contained in:
commit
d8217a6c58
27 changed files with 5332 additions and 40 deletions
2
Gemfile
2
Gemfile
|
@ -113,6 +113,8 @@ gem 'webpacker', '>= 4.0.x'
|
||||||
|
|
||||||
gem 'after_party'
|
gem 'after_party'
|
||||||
|
|
||||||
|
gem 'zxcvbn-ruby', require: 'zxcvbn'
|
||||||
|
|
||||||
# Cron jobs
|
# Cron jobs
|
||||||
gem 'delayed_job_active_record'
|
gem 'delayed_job_active_record'
|
||||||
gem "daemons"
|
gem "daemons"
|
||||||
|
|
|
@ -799,6 +799,7 @@ GEM
|
||||||
nokogiri (~> 1.8)
|
nokogiri (~> 1.8)
|
||||||
xray-rails (0.3.1)
|
xray-rails (0.3.1)
|
||||||
rails (>= 3.1.0)
|
rails (>= 3.1.0)
|
||||||
|
zxcvbn-ruby (0.1.2)
|
||||||
|
|
||||||
PLATFORMS
|
PLATFORMS
|
||||||
ruby
|
ruby
|
||||||
|
@ -901,6 +902,7 @@ DEPENDENCIES
|
||||||
webmock
|
webmock
|
||||||
webpacker (>= 4.0.x)
|
webpacker (>= 4.0.x)
|
||||||
xray-rails
|
xray-rails
|
||||||
|
zxcvbn-ruby
|
||||||
|
|
||||||
BUNDLED WITH
|
BUNDLED WITH
|
||||||
1.16.4
|
1.16.4
|
||||||
|
|
|
@ -0,0 +1,4 @@
|
||||||
|
.administrateurs-activate.container {
|
||||||
|
margin-top: 60px;
|
||||||
|
margin-bottom: 60px;
|
||||||
|
}
|
|
@ -45,6 +45,14 @@ a {
|
||||||
color: $blue;
|
color: $blue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
em {
|
||||||
|
font-style: italic;
|
||||||
|
}
|
||||||
|
|
||||||
|
strong {
|
||||||
|
font-weight: bold;
|
||||||
|
}
|
||||||
|
|
||||||
.container {
|
.container {
|
||||||
@include horizontal-padding($default-padding);
|
@include horizontal-padding($default-padding);
|
||||||
max-width: $page-width + 2 * $default-padding;
|
max-width: $page-width + 2 * $default-padding;
|
||||||
|
|
|
@ -40,3 +40,8 @@
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.one-column-centered {
|
||||||
|
margin: auto;
|
||||||
|
max-width: $page-width / 2;
|
||||||
|
}
|
||||||
|
|
34
app/assets/stylesheets/new_design/password_strength.scss
Normal file
34
app/assets/stylesheets/new_design/password_strength.scss
Normal file
|
@ -0,0 +1,34 @@
|
||||||
|
@import "colors";
|
||||||
|
@import "constants";
|
||||||
|
|
||||||
|
$strength-bg: #EEEEEE;
|
||||||
|
$weak-strength-color: $lighter-red;
|
||||||
|
$medium-strength-color: $orange;
|
||||||
|
$strong-strength-color: $green;
|
||||||
|
|
||||||
|
.password-strength {
|
||||||
|
margin-top: -24px;
|
||||||
|
width: 100%;
|
||||||
|
min-height: 22px;
|
||||||
|
background: $strength-bg;
|
||||||
|
display: block;
|
||||||
|
margin-bottom: $default-spacer;
|
||||||
|
text-align: center;
|
||||||
|
|
||||||
|
&.strength-1 {
|
||||||
|
background: linear-gradient(to right, $weak-strength-color 0%, $weak-strength-color 25%, $strength-bg 25%, $strength-bg 100%);
|
||||||
|
}
|
||||||
|
|
||||||
|
&.strength-2 {
|
||||||
|
background: linear-gradient(to right, $medium-strength-color 0%, $medium-strength-color 50%, $strength-bg 50%, $strength-bg 100%);
|
||||||
|
}
|
||||||
|
|
||||||
|
&.strength-3 {
|
||||||
|
background: linear-gradient(to right, $medium-strength-color 0%, $medium-strength-color 75%, $strength-bg 75%, $strength-bg 100%);
|
||||||
|
}
|
||||||
|
|
||||||
|
&.strength-4 {
|
||||||
|
background: $strong-strength-color;
|
||||||
|
color: #FFFFFF;
|
||||||
|
}
|
||||||
|
}
|
|
@ -16,9 +16,12 @@ class Admin::TypesDeChampController < AdminController
|
||||||
end
|
end
|
||||||
|
|
||||||
def update
|
def update
|
||||||
@procedure.update(TypesDeChampService.create_update_procedure_params params)
|
if @procedure.update(TypesDeChampService.create_update_procedure_params params)
|
||||||
|
flash.now.notice = 'Modifications sauvegardées'
|
||||||
|
else
|
||||||
|
flash.now.alert = @procedure.errors.full_messages.join(', ')
|
||||||
|
end
|
||||||
create_facade
|
create_facade
|
||||||
flash.now.notice = 'Modifications sauvegardées'
|
|
||||||
render 'show', format: :js
|
render 'show', format: :js
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|
|
@ -1,3 +1,5 @@
|
||||||
|
require 'zxcvbn'
|
||||||
|
|
||||||
class Administrateurs::ActivateController < ApplicationController
|
class Administrateurs::ActivateController < ApplicationController
|
||||||
layout "new_application"
|
layout "new_application"
|
||||||
|
|
||||||
|
@ -29,6 +31,11 @@ class Administrateurs::ActivateController < ApplicationController
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def test_password_strength
|
||||||
|
score = Zxcvbn.test(params[:password], [], ZXCVBN_DICTIONNARIES).score
|
||||||
|
render json: { score: score }
|
||||||
|
end
|
||||||
|
|
||||||
private
|
private
|
||||||
|
|
||||||
def update_administrateur_params
|
def update_administrateur_params
|
||||||
|
|
34
app/javascript/new_design/password-strength.js
Normal file
34
app/javascript/new_design/password-strength.js
Normal file
|
@ -0,0 +1,34 @@
|
||||||
|
import $ from 'jquery';
|
||||||
|
|
||||||
|
export function displayPasswordStrength(strengthBarId, score) {
|
||||||
|
var $bar = $('#' + strengthBarId),
|
||||||
|
passwordMessage;
|
||||||
|
|
||||||
|
$bar.removeClass('strength-1 strength-2 strength-3 strength-4');
|
||||||
|
|
||||||
|
if (score < 4) {
|
||||||
|
passwordMessage = 'Mot de passe pas assez complexe';
|
||||||
|
} else {
|
||||||
|
passwordMessage = 'Mot de passe suffisamment complexe';
|
||||||
|
}
|
||||||
|
|
||||||
|
$bar.text(passwordMessage);
|
||||||
|
$bar.addClass('strength-' + score);
|
||||||
|
}
|
||||||
|
|
||||||
|
export function checkPasswordStrength(event, strengthBarId) {
|
||||||
|
var $target = $(event.target),
|
||||||
|
password = $target.val();
|
||||||
|
|
||||||
|
if (password.length > 2) {
|
||||||
|
$.post(
|
||||||
|
'/admin/activate/test_password_strength',
|
||||||
|
{ password: password },
|
||||||
|
function(data) {
|
||||||
|
displayPasswordStrength(strengthBarId, data.score);
|
||||||
|
}
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
displayPasswordStrength(strengthBarId, 0);
|
||||||
|
}
|
||||||
|
}
|
|
@ -24,6 +24,10 @@ import { toggleCondidentielExplanation } from '../new_design/avis';
|
||||||
import { togglePrintMenu } from '../new_design/dossier';
|
import { togglePrintMenu } from '../new_design/dossier';
|
||||||
import { toggleHeaderMenu } from '../new_design/header';
|
import { toggleHeaderMenu } from '../new_design/header';
|
||||||
import { scrollMessagerie } from '../new_design/messagerie';
|
import { scrollMessagerie } from '../new_design/messagerie';
|
||||||
|
import {
|
||||||
|
checkPasswordStrength,
|
||||||
|
displayPasswordStrength
|
||||||
|
} from '../new_design/password-strength';
|
||||||
import { showMotivation, motivationCancel } from '../new_design/state-button';
|
import { showMotivation, motivationCancel } from '../new_design/state-button';
|
||||||
import { toggleChart } from '../new_design/toggle-chart';
|
import { toggleChart } from '../new_design/toggle-chart';
|
||||||
|
|
||||||
|
@ -33,6 +37,8 @@ const DS = {
|
||||||
togglePrintMenu,
|
togglePrintMenu,
|
||||||
toggleHeaderMenu,
|
toggleHeaderMenu,
|
||||||
scrollMessagerie,
|
scrollMessagerie,
|
||||||
|
checkPasswordStrength,
|
||||||
|
displayPasswordStrength,
|
||||||
showMotivation,
|
showMotivation,
|
||||||
motivationCancel,
|
motivationCancel,
|
||||||
toggleChart
|
toggleChart
|
||||||
|
|
|
@ -17,6 +17,17 @@ class Administrateur < ApplicationRecord
|
||||||
|
|
||||||
scope :inactive, -> { where(active: false) }
|
scope :inactive, -> { where(active: false) }
|
||||||
|
|
||||||
|
validate :password_complexity, if: Proc.new { |a| Devise.password_length.include?(a.password.try(:size)) }
|
||||||
|
|
||||||
|
def password_complexity
|
||||||
|
if password.present?
|
||||||
|
score = Zxcvbn.test(password, [], ZXCVBN_DICTIONNARIES).score
|
||||||
|
if score < 4
|
||||||
|
errors.add(:password, :not_strength)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
def self.find_inactive_by_token(reset_password_token)
|
def self.find_inactive_by_token(reset_password_token)
|
||||||
self.inactive.with_reset_password_token(reset_password_token)
|
self.inactive.with_reset_password_token(reset_password_token)
|
||||||
end
|
end
|
||||||
|
|
|
@ -59,6 +59,17 @@ class TypeDeChamp < ApplicationRecord
|
||||||
before_validation :check_mandatory
|
before_validation :check_mandatory
|
||||||
before_save :remove_piece_justificative_template, if: -> { type_champ_changed? }
|
before_save :remove_piece_justificative_template, if: -> { type_champ_changed? }
|
||||||
|
|
||||||
|
def valid?(context = nil)
|
||||||
|
super
|
||||||
|
if dynamic_type.present?
|
||||||
|
dynamic_type.valid?
|
||||||
|
errors.merge!(dynamic_type.errors)
|
||||||
|
end
|
||||||
|
errors.empty?
|
||||||
|
end
|
||||||
|
|
||||||
|
alias_method :validate, :valid?
|
||||||
|
|
||||||
def set_dynamic_type
|
def set_dynamic_type
|
||||||
@dynamic_type = type_champ.present? ? self.class.type_champ_to_class_name(type_champ).constantize.new(self) : nil
|
@dynamic_type = type_champ.present? ? self.class.type_champ_to_class_name(type_champ).constantize.new(self) : nil
|
||||||
end
|
end
|
||||||
|
|
|
@ -3,6 +3,8 @@ class TypesDeChamp::LinkedDropDownListTypeDeChamp < TypesDeChamp::TypeDeChampBas
|
||||||
|
|
||||||
delegate :drop_down_list, to: :@type_de_champ
|
delegate :drop_down_list, to: :@type_de_champ
|
||||||
|
|
||||||
|
validate :check_presence_of_primary_options
|
||||||
|
|
||||||
def primary_options
|
def primary_options
|
||||||
primary_options = unpack_options.map(&:first)
|
primary_options = unpack_options.map(&:first)
|
||||||
if primary_options.present?
|
if primary_options.present?
|
||||||
|
@ -21,13 +23,19 @@ class TypesDeChamp::LinkedDropDownListTypeDeChamp < TypesDeChamp::TypeDeChampBas
|
||||||
|
|
||||||
private
|
private
|
||||||
|
|
||||||
|
def check_presence_of_primary_options
|
||||||
|
if !PRIMARY_PATTERN.match?(drop_down_list.options.second)
|
||||||
|
errors.add(libelle, "doit commencer par une entrée de menu primaire de la forme <code style='white-space: pre-wrap;'>--texte--</code>")
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
def unpack_options
|
def unpack_options
|
||||||
_, *options = drop_down_list.options
|
_, *options = drop_down_list.options
|
||||||
chunked = options.slice_before(PRIMARY_PATTERN)
|
chunked = options.slice_before(PRIMARY_PATTERN)
|
||||||
chunked.map do |chunk|
|
chunked.map do |chunk|
|
||||||
primary, *secondary = chunk
|
primary, *secondary = chunk
|
||||||
secondary.unshift('')
|
secondary.unshift('')
|
||||||
[PRIMARY_PATTERN.match(primary)[1], secondary]
|
[PRIMARY_PATTERN.match(primary)&.[](1), secondary]
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -1,4 +1,8 @@
|
||||||
class TypesDeChamp::TypeDeChampBase
|
class TypesDeChamp::TypeDeChampBase
|
||||||
|
include ActiveModel::Validations
|
||||||
|
|
||||||
|
delegate :libelle, to: :@type_de_champ
|
||||||
|
|
||||||
def initialize(type_de_champ)
|
def initialize(type_de_champ)
|
||||||
@type_de_champ = type_de_champ
|
@type_de_champ = type_de_champ
|
||||||
end
|
end
|
||||||
|
|
|
@ -1,8 +1,29 @@
|
||||||
.container
|
- content_for(:title, "Choix du mot de passe")
|
||||||
= form_for @administrateur, url: { controller: 'administrateurs/activate', action: :create }, html: { class: "form" } do |f|
|
|
||||||
%br
|
- content_for :footer do
|
||||||
%h1
|
= render partial: "root/footer"
|
||||||
= @administrateur.email
|
|
||||||
= f.password_field :password, placeholder: 'Mot de passe'
|
.administrateurs-activate.container
|
||||||
= f.hidden_field :reset_password_token, value: params[:token]
|
.one-column-centered
|
||||||
= f.submit 'Définir le mot de passe', class: 'button large primary expand'
|
= form_for @administrateur, url: { controller: 'administrateurs/activate', action: :create }, html: { class: "form" } do |f|
|
||||||
|
%br
|
||||||
|
%h1
|
||||||
|
Choix du mot de passe
|
||||||
|
|
||||||
|
= f.label :email, "Email"
|
||||||
|
= f.text_field :email, disabled: true
|
||||||
|
|
||||||
|
= f.label :password do
|
||||||
|
Mot de passe
|
||||||
|
|
||||||
|
= f.password_field :password, placeholder: 'Mot de passe', onKeyUp: "javascript:DS.checkPasswordStrength(event, 'strength-bar', 'submit-password');"
|
||||||
|
|
||||||
|
#strength-bar.password-strength
|
||||||
|
|
||||||
|
|
||||||
|
.explication
|
||||||
|
%strong Aide :
|
||||||
|
Une courte phrase peut être un mot de passe très sécurisé.
|
||||||
|
|
||||||
|
= f.hidden_field :reset_password_token, value: params[:token]
|
||||||
|
= f.submit 'Continuer', class: 'button large primary expand', id: "submit-password", data: { disable_with: "Envoi..." }
|
||||||
|
|
1
config/initializers/zxcvbn.rb
Normal file
1
config/initializers/zxcvbn.rb
Normal file
|
@ -0,0 +1 @@
|
||||||
|
ZXCVBN_DICTIONNARIES = YAML.safe_load(File.read(Rails.root.join("config", "initializers", "zxcvbn_dictionnaries.yaml")))
|
5002
config/initializers/zxcvbn_dictionnaries.yaml
Normal file
5002
config/initializers/zxcvbn_dictionnaries.yaml
Normal file
File diff suppressed because it is too large
Load diff
15
config/locales/models/administrateur/fr.yml
Normal file
15
config/locales/models/administrateur/fr.yml
Normal file
|
@ -0,0 +1,15 @@
|
||||||
|
fr:
|
||||||
|
activerecord:
|
||||||
|
attributes:
|
||||||
|
administrateur:
|
||||||
|
password: Le mot de passe
|
||||||
|
errors:
|
||||||
|
models:
|
||||||
|
administrateur:
|
||||||
|
attributes:
|
||||||
|
email:
|
||||||
|
blank: 'doit être rempli'
|
||||||
|
password:
|
||||||
|
too_short: 'est trop court'
|
||||||
|
blank: 'doit être rempli'
|
||||||
|
not_strength: "n'est pas assez complexe"
|
|
@ -175,6 +175,7 @@ Rails.application.routes.draw do
|
||||||
namespace :admin do
|
namespace :admin do
|
||||||
get 'activate' => '/administrateurs/activate#new'
|
get 'activate' => '/administrateurs/activate#new'
|
||||||
patch 'activate' => '/administrateurs/activate#create'
|
patch 'activate' => '/administrateurs/activate#create'
|
||||||
|
post 'activate/test_password_strength' => '/administrateurs/activate#test_password_strength'
|
||||||
get 'sign_in' => '/administrateurs/sessions#new'
|
get 'sign_in' => '/administrateurs/sessions#new'
|
||||||
get 'procedures/archived' => 'procedures#archived'
|
get 'procedures/archived' => 'procedures#archived'
|
||||||
get 'procedures/draft' => 'procedures#draft'
|
get 'procedures/draft' => 'procedures#draft'
|
||||||
|
|
|
@ -7,9 +7,9 @@ describe Gestionnaires::PasswordsController, type: :controller do
|
||||||
|
|
||||||
describe "update" do
|
describe "update" do
|
||||||
context "unified login" do
|
context "unified login" do
|
||||||
let(:gestionnaire) { create(:gestionnaire, email: 'unique@plop.com', password: 'password') }
|
let(:gestionnaire) { create(:gestionnaire, email: 'unique@plop.com', password: 'un super mot de passe') }
|
||||||
let(:user) { create(:user, email: 'unique@plop.com', password: 'password') }
|
let(:user) { create(:user, email: 'unique@plop.com', password: 'un super mot de passe') }
|
||||||
let(:administrateur) { create(:administrateur, email: 'unique@plop.com', password: 'password') }
|
let(:administrateur) { create(:administrateur, email: 'unique@plop.com', password: 'un super mot de passe') }
|
||||||
|
|
||||||
before do
|
before do
|
||||||
@token = gestionnaire.send(:set_reset_password_token)
|
@token = gestionnaire.send(:set_reset_password_token)
|
||||||
|
|
|
@ -7,9 +7,9 @@ describe Users::PasswordsController, type: :controller do
|
||||||
|
|
||||||
describe "update" do
|
describe "update" do
|
||||||
context "unified login" do
|
context "unified login" do
|
||||||
let(:user) { create(:user, email: 'unique@plop.com', password: 'password') }
|
let(:user) { create(:user, email: 'unique@plop.com', password: 'mot de passe complexe') }
|
||||||
let(:gestionnaire) { create(:gestionnaire, email: 'unique@plop.com', password: 'password') }
|
let(:gestionnaire) { create(:gestionnaire, email: 'unique@plop.com', password: 'mot de passe complexe') }
|
||||||
let(:administrateur) { create(:administrateur, email: 'unique@plop.com', password: 'password') }
|
let(:administrateur) { create(:administrateur, email: 'unique@plop.com', password: 'mot de passe complexe') }
|
||||||
|
|
||||||
before do
|
before do
|
||||||
@token = user.send(:set_reset_password_token)
|
@token = user.send(:set_reset_password_token)
|
||||||
|
@ -21,8 +21,8 @@ describe Users::PasswordsController, type: :controller do
|
||||||
put :update, params: {
|
put :update, params: {
|
||||||
user: {
|
user: {
|
||||||
reset_password_token: @token,
|
reset_password_token: @token,
|
||||||
password: "supersecret",
|
password: "mot de passe super secret",
|
||||||
password_confirmation: "supersecret",
|
password_confirmation: "mot de passe super secret",
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
expect(subject.current_user).to eq(user)
|
expect(subject.current_user).to eq(user)
|
||||||
|
@ -33,8 +33,8 @@ describe Users::PasswordsController, type: :controller do
|
||||||
put :update, params: {
|
put :update, params: {
|
||||||
user: {
|
user: {
|
||||||
reset_password_token: @token,
|
reset_password_token: @token,
|
||||||
password: "supersecret",
|
password: "mot de passe super secret",
|
||||||
password_confirmation: "supersecret",
|
password_confirmation: "mot de passe super secret",
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
expect(subject.current_user).to eq(user)
|
expect(subject.current_user).to eq(user)
|
||||||
|
|
|
@ -24,7 +24,7 @@ describe Users::SessionsController, type: :controller do
|
||||||
|
|
||||||
context "unified login" do
|
context "unified login" do
|
||||||
let(:email) { 'unique@plop.com' }
|
let(:email) { 'unique@plop.com' }
|
||||||
let(:password) { 'password' }
|
let(:password) { 'un super mot de passe' }
|
||||||
|
|
||||||
let(:user) { create(:user, email: email, password: password) }
|
let(:user) { create(:user, email: email, password: password) }
|
||||||
let(:gestionnaire) { create(:gestionnaire, email: email, password: password) }
|
let(:gestionnaire) { create(:gestionnaire, email: email, password: password) }
|
||||||
|
@ -80,8 +80,8 @@ describe Users::SessionsController, type: :controller do
|
||||||
end
|
end
|
||||||
|
|
||||||
context 'with different passwords' do
|
context 'with different passwords' do
|
||||||
let!(:gestionnaire) { create(:gestionnaire, email: email, password: 'another_password') }
|
let!(:gestionnaire) { create(:gestionnaire, email: email, password: 'mot de passe complexe') }
|
||||||
let!(:administrateur) { create(:administrateur, email: email, password: 'another_password') }
|
let!(:administrateur) { create(:administrateur, email: email, password: 'mot de passe complexe') }
|
||||||
|
|
||||||
before do
|
before do
|
||||||
user
|
user
|
||||||
|
@ -165,7 +165,7 @@ describe Users::SessionsController, type: :controller do
|
||||||
end
|
end
|
||||||
|
|
||||||
context "when associated administrateur" do
|
context "when associated administrateur" do
|
||||||
let(:administrateur) { create(:administrateur, email: 'unique@plop.com', password: 'password') }
|
let(:administrateur) { create(:administrateur, email: 'unique@plop.com') }
|
||||||
|
|
||||||
it 'signs user + gestionnaire + administrateur out' do
|
it 'signs user + gestionnaire + administrateur out' do
|
||||||
sign_in user
|
sign_in user
|
||||||
|
|
|
@ -2,6 +2,6 @@ FactoryBot.define do
|
||||||
sequence(:administrateur_email) { |n| "admin#{n}@admin.com" }
|
sequence(:administrateur_email) { |n| "admin#{n}@admin.com" }
|
||||||
factory :administrateur do
|
factory :administrateur do
|
||||||
email { generate(:administrateur_email) }
|
email { generate(:administrateur_email) }
|
||||||
password { 'password' }
|
password { 'mon chien aime les bananes' }
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -54,7 +54,7 @@ FactoryBot.define do
|
||||||
end
|
end
|
||||||
factory :type_de_champ_linked_drop_down_list do
|
factory :type_de_champ_linked_drop_down_list do
|
||||||
type_champ { TypeDeChamp.type_champs.fetch(:linked_drop_down_list) }
|
type_champ { TypeDeChamp.type_champs.fetch(:linked_drop_down_list) }
|
||||||
drop_down_list { create(:drop_down_list) }
|
drop_down_list { create(:drop_down_list, value: "--primary--\nsecondary\n") }
|
||||||
end
|
end
|
||||||
factory :type_de_champ_pays do
|
factory :type_de_champ_pays do
|
||||||
type_champ { TypeDeChamp.type_champs.fetch(:pays) }
|
type_champ { TypeDeChamp.type_champs.fetch(:pays) }
|
||||||
|
|
|
@ -9,7 +9,7 @@ describe Administrateur, type: :model do
|
||||||
end
|
end
|
||||||
|
|
||||||
describe 'after_save' do
|
describe 'after_save' do
|
||||||
subject { described_class.new(email: 'toto@tps.com', password: 'password') }
|
subject { create(:administrateur) }
|
||||||
before do
|
before do
|
||||||
subject.save
|
subject.save
|
||||||
end
|
end
|
||||||
|
@ -19,7 +19,7 @@ describe Administrateur, type: :model do
|
||||||
describe 'generate_api_token' do
|
describe 'generate_api_token' do
|
||||||
let(:token) { 'bullshit' }
|
let(:token) { 'bullshit' }
|
||||||
let(:new_token) { 'pocket_master' }
|
let(:new_token) { 'pocket_master' }
|
||||||
let!(:admin_1) { create(:administrateur, email: 'toto@tps.com', password: 'password', api_token: token) }
|
let!(:admin_1) { create(:administrateur, api_token: token) }
|
||||||
before do
|
before do
|
||||||
allow(SecureRandom).to receive(:hex).and_return(token, new_token)
|
allow(SecureRandom).to receive(:hex).and_return(token, new_token)
|
||||||
admin_1.renew_api_token
|
admin_1.renew_api_token
|
||||||
|
@ -34,22 +34,22 @@ describe Administrateur, type: :model do
|
||||||
administrateur = create(:administrateur)
|
administrateur = create(:administrateur)
|
||||||
user = create(:user, email: administrateur.email)
|
user = create(:user, email: administrateur.email)
|
||||||
|
|
||||||
administrateur.update(email: 'whoami@plop.com', password: 'super secret')
|
administrateur.update(email: 'whoami@plop.com', password: 'voilà un super mdp')
|
||||||
|
|
||||||
user.reload
|
user.reload
|
||||||
expect(user.email).to eq('whoami@plop.com')
|
expect(user.email).to eq('whoami@plop.com')
|
||||||
expect(user.valid_password?('super secret')).to be(true)
|
expect(user.valid_password?('voilà un super mdp')).to be(true)
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'syncs credentials to associated administrateur' do
|
it 'syncs credentials to associated administrateur' do
|
||||||
administrateur = create(:administrateur)
|
administrateur = create(:administrateur)
|
||||||
gestionnaire = create(:gestionnaire, email: administrateur.email)
|
gestionnaire = create(:gestionnaire, email: administrateur.email)
|
||||||
|
|
||||||
administrateur.update(email: 'whoami@plop.com', password: 'super secret')
|
administrateur.update(email: 'whoami@plop.com', password: 'et encore un autre mdp')
|
||||||
|
|
||||||
gestionnaire.reload
|
gestionnaire.reload
|
||||||
expect(gestionnaire.email).to eq('whoami@plop.com')
|
expect(gestionnaire.email).to eq('whoami@plop.com')
|
||||||
expect(gestionnaire.valid_password?('super secret')).to be(true)
|
expect(gestionnaire.valid_password?('et encore un autre mdp')).to be(true)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@ -64,8 +64,8 @@ describe Administrateur, type: :model do
|
||||||
let(:administrateur) { create(:administration).invite_admin('paul@tps.fr') }
|
let(:administrateur) { create(:administration).invite_admin('paul@tps.fr') }
|
||||||
let(:reset_password_token) { administrateur.invite!(administration.id) }
|
let(:reset_password_token) { administrateur.invite!(administration.id) }
|
||||||
|
|
||||||
it { expect(Administrateur.reset_password(reset_password_token, '12345678').errors).to be_empty }
|
it { expect(Administrateur.reset_password(reset_password_token, "j'aime manger des radis").errors).to be_empty }
|
||||||
it { expect(Administrateur.reset_password('123', '12345678').errors).not_to be_empty }
|
it { expect(Administrateur.reset_password('123', "j'aime manger des radis").errors).not_to be_empty }
|
||||||
it { expect(Administrateur.reset_password(reset_password_token, '').errors).not_to be_empty }
|
it { expect(Administrateur.reset_password(reset_password_token, '').errors).not_to be_empty }
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@ -79,4 +79,30 @@ describe Administrateur, type: :model do
|
||||||
it { expect(administrateur.feature_enabled?(:champ_siret)).to be_falsey }
|
it { expect(administrateur.feature_enabled?(:champ_siret)).to be_falsey }
|
||||||
it { expect(administrateur.feature_enabled?(:champ_pj)).to be_truthy }
|
it { expect(administrateur.feature_enabled?(:champ_pj)).to be_truthy }
|
||||||
end
|
end
|
||||||
|
|
||||||
|
describe "#password_complexity" do
|
||||||
|
let(:administrateur) { build(:administrateur, password: password) }
|
||||||
|
|
||||||
|
subject do
|
||||||
|
administrateur.save
|
||||||
|
administrateur.errors[:password]
|
||||||
|
end
|
||||||
|
|
||||||
|
context "with a strong password" do
|
||||||
|
let(:password) { "la démat c'est simple" }
|
||||||
|
it { expect(subject).to eq([]) }
|
||||||
|
end
|
||||||
|
|
||||||
|
context "with a weak password" do
|
||||||
|
let(:password) { "12345678" }
|
||||||
|
it { expect(subject).to include "n'est pas assez complexe" }
|
||||||
|
it { expect(subject).not_to include "est trop court" }
|
||||||
|
end
|
||||||
|
|
||||||
|
context "with a short password" do
|
||||||
|
let(:password) { "1" }
|
||||||
|
it { expect(subject).to include "est trop court" }
|
||||||
|
it { expect(subject).not_to include "n'est pas assez complexe" }
|
||||||
|
end
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -79,5 +79,26 @@ shared_examples 'type_de_champ_spec' do
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
context 'delegate validation to dynamic type' do
|
||||||
|
subject { build(:type_de_champ_text) }
|
||||||
|
let(:dynamic_type) do
|
||||||
|
Class.new(TypesDeChamp::TypeDeChampBase) do
|
||||||
|
validate :never_valid
|
||||||
|
|
||||||
|
def never_valid
|
||||||
|
errors.add(:troll, 'always invalid')
|
||||||
|
end
|
||||||
|
end.new(subject)
|
||||||
|
end
|
||||||
|
|
||||||
|
before { subject.instance_variable_set(:@dynamic_type, dynamic_type) }
|
||||||
|
|
||||||
|
it { is_expected.to be_invalid }
|
||||||
|
it do
|
||||||
|
subject.validate
|
||||||
|
expect(subject.errors.full_messages.to_sentence).to eq('Troll always invalid')
|
||||||
|
end
|
||||||
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -1,12 +1,78 @@
|
||||||
require 'spec_helper'
|
require 'spec_helper'
|
||||||
|
|
||||||
describe TypesDeChamp::LinkedDropDownListTypeDeChamp do
|
describe TypesDeChamp::LinkedDropDownListTypeDeChamp do
|
||||||
|
let(:drop_down_list) { build(:drop_down_list, value: menu_options) }
|
||||||
|
let(:type_de_champ) { build(:type_de_champ_linked_drop_down_list, drop_down_list: drop_down_list) }
|
||||||
|
|
||||||
|
subject { type_de_champ.dynamic_type }
|
||||||
|
|
||||||
|
describe 'validation' do
|
||||||
|
context 'It must start with one primary option' do
|
||||||
|
context 'valid menu' do
|
||||||
|
let(:menu_options) do
|
||||||
|
<<~END_OPTIONS
|
||||||
|
--Primary 1--
|
||||||
|
secondary 1.1
|
||||||
|
secondary 1.2
|
||||||
|
--Primary 2--
|
||||||
|
secondary 2.1
|
||||||
|
secondary 2.2
|
||||||
|
secondary 2.3
|
||||||
|
END_OPTIONS
|
||||||
|
end
|
||||||
|
|
||||||
|
it { is_expected.to be_valid }
|
||||||
|
end
|
||||||
|
|
||||||
|
context 'degenerate but valid menu' do
|
||||||
|
let(:menu_options) do
|
||||||
|
<<~END_OPTIONS
|
||||||
|
--Primary 1--
|
||||||
|
END_OPTIONS
|
||||||
|
end
|
||||||
|
|
||||||
|
it { is_expected.to be_valid }
|
||||||
|
end
|
||||||
|
|
||||||
|
context 'invalid menus' do
|
||||||
|
shared_examples 'missing primary option' do
|
||||||
|
it { is_expected.to be_invalid }
|
||||||
|
it do
|
||||||
|
subject.validate
|
||||||
|
expect(subject.errors.full_messages).to eq ["#{subject.libelle} doit commencer par une entrée de menu primaire de la forme <code style='white-space: pre-wrap;'>--texte--</code>"]
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
context 'no primary option' do
|
||||||
|
let(:menu_options) do
|
||||||
|
<<~END_OPTIONS
|
||||||
|
secondary 1.1
|
||||||
|
secondary 1.2
|
||||||
|
END_OPTIONS
|
||||||
|
end
|
||||||
|
|
||||||
|
it_should_behave_like 'missing primary option'
|
||||||
|
end
|
||||||
|
|
||||||
|
context 'starting with secondary options' do
|
||||||
|
let(:menu_options) do
|
||||||
|
<<~END_OPTIONS
|
||||||
|
secondary 1.1
|
||||||
|
secondary 1.2
|
||||||
|
--Primary 2--
|
||||||
|
secondary 2.1
|
||||||
|
secondary 2.2
|
||||||
|
secondary 2.3
|
||||||
|
END_OPTIONS
|
||||||
|
end
|
||||||
|
|
||||||
|
it_should_behave_like 'missing primary option'
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
describe '#unpack_options' do
|
describe '#unpack_options' do
|
||||||
let(:drop_down_list) { build(:drop_down_list, value: menu_options) }
|
|
||||||
let(:type_de_champ) { build(:type_de_champ_linked_drop_down_list, drop_down_list: drop_down_list) }
|
|
||||||
|
|
||||||
subject { type_de_champ.dynamic_type }
|
|
||||||
|
|
||||||
context 'with no options' do
|
context 'with no options' do
|
||||||
let(:menu_options) { '' }
|
let(:menu_options) { '' }
|
||||||
it { expect(subject.secondary_options).to eq({}) }
|
it { expect(subject.secondary_options).to eq({}) }
|
||||||
|
|
Loading…
Reference in a new issue