Merge pull request #6742 from betagouv/main

This commit is contained in:
Pierre de La Morinerie 2021-12-09 13:11:33 +01:00 committed by GitHub
commit 14af63d32f
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
49 changed files with 979 additions and 75 deletions

View file

@ -358,7 +358,8 @@
}
.cnaf-inputs,
.dgfip-inputs {
.dgfip-inputs,
.pole-emploi-inputs {
display: flex;
flex-wrap: wrap;
justify-content: space-between;

View file

@ -0,0 +1,30 @@
@import "constants";
@import "colors";
table.pole-emploi {
margin: 2 * $default-padding 0 $default-padding $default-padding;
width: 100%;
caption {
font-weight: bold;
margin-left: - $default-padding;
margin-bottom: $default-spacer;
text-align: left;
}
th,
td {
font-weight: normal;
padding: $default-spacer;
}
th.text-right {
text-align: right;
}
&.horizontal {
th {
border-bottom: 1px solid $grey;
}
}
}

View file

@ -235,7 +235,7 @@ module Instructeurs
flash.notice = 'Le dossier a bien été supprimé'
redirect_to instructeur_procedure_path(procedure)
else
flash.alert = "Suppression impossible : le dossier nest pas terminé"
flash.alert = "Suppression impossible : le dossier nest pas traité"
redirect_back(fallback_location: instructeur_procedures_url)
end
end

View file

@ -217,7 +217,7 @@ module Users
if dossier.can_be_deleted_by_user?
dossier.discard_and_keep_track!(current_user, :user_request)
flash.notice = t('.deleted_dossier')
flash.notice = t('.soft_deleted_dossier')
redirect_to dossiers_path
else
flash.notice = t('.undergoingreview')
@ -291,7 +291,7 @@ module Users
def hide_dossier
dossier = current_user.dossiers.includes(:user, procedure: :administrateurs).find(params[:id])
dossier.update(hidden_by_user_at: Time.zone.now)
flash.notice = t('users.dossiers.ask_deletion.deleted_dossier')
flash.notice = t('users.dossiers.ask_deletion.soft_deleted_dossier')
redirect_to dossiers_path
end
@ -348,8 +348,8 @@ module Users
def champs_params
params.permit(dossier: {
champs_attributes: [
:id, :value, :value_other, :external_id, :primary_value, :secondary_value, :numero_allocataire, :code_postal, :numero_fiscal, :reference_avis, :piece_justificative_file, :departement, :code_departement, value: [],
champs_attributes: [:id, :_destroy, :value, :value_other, :external_id, :primary_value, :secondary_value, :numero_allocataire, :code_postal, :numero_fiscal, :reference_avis, :piece_justificative_file, :departement, :code_departement, value: []]
:id, :value, :value_other, :external_id, :primary_value, :secondary_value, :numero_allocataire, :code_postal, :identifiant, :numero_fiscal, :reference_avis, :piece_justificative_file, :departement, :code_departement, value: [],
champs_attributes: [:id, :_destroy, :value, :value_other, :external_id, :primary_value, :secondary_value, :numero_allocataire, :code_postal, :identifiant, :numero_fiscal, :reference_avis, :piece_justificative_file, :departement, :code_departement, value: []]
]
})
end

View file

@ -2008,6 +2008,11 @@ enum TypeDeChamp {
"""
piece_justificative
"""
Situation Pôle emploi
"""
pole_emploi
"""
Régions
"""

View file

@ -143,8 +143,10 @@ module Types
end
def attestation
if object.termine? && object.procedure.attestation_template&.activated?
Loaders::Association.for(object.class, attestation: { pdf_attachment: :blob }).load(object).then(&:pdf)
if object.attestation_activated?
Loaders::Association.for(object.class, attestation: { pdf_attachment: :blob })
.load(object)
.then { |attestation| attestation&.pdf }
end
end

View file

@ -4,6 +4,7 @@ class APIParticulier::API
INTROSPECT_RESOURCE_NAME = "introspect"
COMPOSITION_FAMILIALE_RESOURCE_NAME = "v2/composition-familiale"
AVIS_IMPOSITION_RESOURCE_NAME = "v2/avis-imposition"
SITUATION_POLE_EMPLOI = "v2/situations-pole-emploi"
TIMEOUT = 20
@ -29,6 +30,10 @@ class APIParticulier::API
referenceAvis: reference_avis.to_i.to_s.rjust(13, "0"))
end
def situation_pole_emploi(identifiant)
get(SITUATION_POLE_EMPLOI, identifiant: identifiant)
end
private
def get(resource_name, params = {})

View file

@ -0,0 +1,46 @@
class APIParticulier::PoleEmploiAdapter
class InvalidSchemaError < ::StandardError
def initialize(errors)
super(errors.map(&:to_json).join("\n"))
end
end
def initialize(api_particulier_token, identifiant, requested_sources)
@api = APIParticulier::API.new(api_particulier_token)
@identifiant = identifiant
@requested_sources = requested_sources
end
def to_params
@api.situation_pole_emploi(@identifiant)
.tap { |d| ensure_valid_schema!(d) }
.then { |d| extract_requested_sources(d) }
end
private
def ensure_valid_schema!(data)
if !schemer.valid?(data)
errors = schemer.validate(data).to_a
raise InvalidSchemaError.new(errors)
end
end
def schemer
@schemer ||= JSONSchemer.schema(Rails.root.join('app/schemas/situation-pole-emploi.json'))
end
def extract_requested_sources(data)
@requested_sources['pole_emploi']&.map do |(scope, sources)|
case scope
when 'adresse'
sources.map { |source| { scope => data[scope].slice(*source) } }
when 'identifiant', 'contact', 'inscription'
sources.map { |source| { scope => data.slice(*source) } }
else
{ scope => data.slice(*sources) }
end
end
&.flatten&.reduce(&:deep_merge) || {}
end
end

View file

@ -7,8 +7,7 @@ module APIParticulier
def available_sources
@procedure.api_particulier_scopes
.map { |provider_and_scope| raw_scopes[provider_and_scope] }
.compact
.filter_map { |provider_and_scope| raw_scopes[provider_and_scope] }
.map { |provider, scope| extract_sources(provider, scope) }
.reduce({}) { |acc, el| acc.deep_merge(el) { |_, this_val, other_val| this_val + other_val } }
end
@ -77,7 +76,11 @@ module APIParticulier
'dgfip_annee_impot' => ['dgfip', 'annee_impot'],
'dgfip_annee_revenus' => ['dgfip', 'annee_revenus'],
'dgfip_erreur_correctif' => ['dgfip', 'erreur_correctif'],
'dgfip_situation_partielle' => ['dgfip', 'situation_partielle']
'dgfip_situation_partielle' => ['dgfip', 'situation_partielle'],
'pole_emploi_identite' => ['pole_emploi', 'identite'],
'pole_emploi_adresse' => ['pole_emploi', 'adresse'],
'pole_emploi_contact' => ['pole_emploi', 'contact'],
'pole_emploi_inscription' => ['pole_emploi', 'inscription']
}
end
@ -114,6 +117,12 @@ module APIParticulier
'annee_revenus' => { 'agregats_fiscaux' => ['anneeRevenus'] },
'erreur_correctif' => { 'complements' => ['erreurCorrectif'] },
'situation_partielle' => { 'complements' => ['situationPartielle'] }
},
'pole_emploi' => {
'identite' => ['identifiant', 'civilite', 'nom', 'nomUsage', 'prenom', 'sexe', 'dateNaissance'],
'adresse' => ['INSEECommune', 'codePostal', 'localite', 'ligneVoie', 'ligneComplementDestinataire', 'ligneComplementAdresse', 'ligneComplementDistribution', 'ligneNom'],
'contact' => ['email', 'telephone', 'telephone2'],
'inscription' => ['dateInscription', 'dateCessationInscription', 'codeCertificationCNAV', 'codeCategorieInscription', 'libelleCategorieInscription']
}
}
end

View file

@ -54,7 +54,11 @@ class Helpscout::FormAdapter
private
def add_tags(conversation_id)
@api.add_tags(conversation_id, params[:tags])
@api.add_tags(conversation_id, tags)
end
def tags
(params[:tags].presence || []) + ['contact form']
end
def create_conversation

View file

@ -0,0 +1,47 @@
# == Schema Information
#
# Table name: champs
#
# id :integer not null, primary key
# data :jsonb
# fetch_external_data_exceptions :string is an Array
# private :boolean default(FALSE), not null
# rebased_at :datetime
# row :integer
# type :string
# value :string
# value_json :jsonb
# created_at :datetime
# updated_at :datetime
# dossier_id :integer
# etablissement_id :integer
# external_id :string
# parent_id :bigint
# type_de_champ_id :integer
#
class Champs::PoleEmploiChamp < Champs::TextChamp
# see https://github.com/betagouv/api-particulier/blob/master/src/presentation/middlewares/pole-emploi-input-validation.middleware.ts
store_accessor :value_json, :identifiant
def blank?
external_id.nil?
end
def fetch_external_data?
true
end
def fetch_external_data
return if !valid?
APIParticulier::PoleEmploiAdapter.new(
procedure.api_particulier_token,
identifiant,
procedure.api_particulier_sources
).to_params
end
def external_id
{ identifiant: identifiant }.to_json if identifiant.present?
end
end

View file

@ -779,6 +779,10 @@ class Dossier < ApplicationRecord
end
end
def attestation_activated?
termine? && procedure.attestation_template&.activated?
end
def after_passer_en_construction
self.conservation_extension = 0.days
self.depose_at = self.en_construction_at = self.traitements

View file

@ -730,6 +730,10 @@ class Procedure < ApplicationRecord
api_particulier_sources['dgfip'].present?
end
def pole_emploi_enabled?
api_particulier_sources['pole_emploi'].present?
end
private
def validate_for_publication?

View file

@ -51,7 +51,8 @@ class TypeDeChamp < ApplicationRecord
iban: 'iban',
annuaire_education: 'annuaire_education',
cnaf: 'cnaf',
dgfip: 'dgfip'
dgfip: 'dgfip',
pole_emploi: 'pole_emploi'
}
belongs_to :revision, class_name: 'ProcedureRevision', optional: true
@ -327,6 +328,8 @@ class TypeDeChamp < ApplicationRecord
procedure.cnaf_enabled?
when TypeDeChamp.type_champs.fetch(:dgfip)
procedure.dgfip_enabled?
when TypeDeChamp.type_champs.fetch(:pole_emploi)
procedure.pole_emploi_enabled?
else
true
end

View file

@ -0,0 +1,2 @@
class TypesDeChamp::PoleEmploiTypeDeChamp < TypesDeChamp::TextTypeDeChamp
end

View file

@ -0,0 +1,87 @@
{
"$schema": "http://json-schema.org/draft-07/schema#",
"$id": "http://demarches-simplifiees.fr/situation-pole-emploi.schema.json",
"title": "situation pole emploi",
"type": "object",
"properties": {
"identifiant": {
"type": "string"
},
"civilite": {
"type": "string"
},
"nom": {
"type": "string"
},
"nomUsage": {
"type": "string"
},
"prenom": {
"type": "string"
},
"sexe": {
"enum": ["F", "M"]
},
"dateNaissance": {
"format": "date",
"type": "string"
},
"codeCertificationCNAV": {
"type": "string"
},
"telephone": {
"type": "string"
},
"telephone2": {
"type": "string"
},
"email": {
"type": "string"
},
"$defs": {
"adresse": {
"type": "object",
"properties": {
"codePostal": {
"type": "string"
},
"INSEECommune": {
"type": "string"
},
"localite": {
"type": "string"
},
"ligneVoie": {
"type": "string"
},
"ligneComplementDestinataire": {
"type": "string"
},
"ligneComplementAdresse": {
"type": "string"
},
"ligneComplementDistribution": {
"type": "string"
},
"ligneNom": {
"type": "string"
}
}
}
},
"dateInscription": {
"format": "date",
"type": "string"
},
"dateCessationInscription": {
"format": "date",
"type": "string"
},
"codeCategorieInscription": {
"type": "integer"
},
"libelleCategroiieInscription": {
"type": "string"
}
}
}

View file

@ -2,7 +2,7 @@ class NotificationService
class << self
def send_instructeur_email_notification
Instructeur
.includes(assign_to: { procedure: :dossiers })
.includes(assign_to: [:procedure])
.where(assign_tos: { daily_email_notifications_enabled: true })
.find_in_batches { |instructeurs| send_batch_of_instructeurs_email_notification(instructeurs) }
end

View file

@ -16,7 +16,7 @@
- scopes.each do |scope_key, sources|
%h3.explication-libelle= t("api_particulier.providers.#{provider_key}.scopes.#{scope_key}.libelle")
%ul.procedure-admin-api-particulier-sources{ id: scope_key }
%ul.procedure-admin-api-particulier-sources{ id: "#{provider_key}-#{scope_key}" }
- sources.each do |source_key, enabled_hash|
- enabled = (@procedure.api_particulier_sources.dig(provider_key, scope_key)&.include?(source_key)).present?
%li

View file

@ -0,0 +1,6 @@
%table.pole-emploi
%caption #{t("api_particulier.providers.pole_emploi.scopes.#{scope}.libelle")} :
- adresse.slice('INSEECommune', 'codePostal', 'localite', 'ligneVoie', 'ligneComplementDestinataire', 'ligneComplementAdresse', 'ligneComplementDistribution', 'ligneNom').keys.each do |key|
%tr
%th= t("api_particulier.providers.pole_emploi.scopes.#{scope}.#{key}")
%td= adresse[key]

View file

@ -0,0 +1,6 @@
%table.pole-emploi
%caption #{t("api_particulier.providers.pole_emploi.scopes.#{scope}.libelle")} :
- contact.slice('email', 'telephone', 'telephone2').keys.each do |key|
%tr
%th= t("api_particulier.providers.pole_emploi.scopes.#{scope}.#{key}")
%td= contact[key]

View file

@ -0,0 +1,12 @@
%table.pole-emploi
%caption #{t("api_particulier.providers.pole_emploi.scopes.#{scope}.libelle")} :
- identite.slice('identifiant', 'civilite', 'nom', 'nomUsage', 'prenom', 'sexe', 'dateNaissance').keys.each do |key|
%tr
%th= t("api_particulier.providers.pole_emploi.scopes.#{scope}.#{key}")
- case key
- when 'dateNaissance'
%td= try_format_date(Date.strptime(identite[key], "%Y-%m-%d"))
- when 'sexe'
%td= t("api_particulier.providers.pole_emploi.scopes.#{scope}.#{identite[key]}")
- else
%td= identite[key]

View file

@ -0,0 +1,10 @@
%table.pole-emploi
%caption #{t("api_particulier.providers.pole_emploi.scopes.#{scope}.libelle")} :
- inscription.slice('dateInscription', 'dateCessationInscription', 'codeCertificationCNAV', 'codeCategorieInscription', 'libelleCategorieInscription').keys.each do |key|
%tr
%th= t("api_particulier.providers.pole_emploi.scopes.#{scope}.#{key}")
- case key
- when 'dateInscription', 'dateCessationInscription'
%td= try_format_date(Date.strptime(inscription[key], "%Y-%m-%d"))
- else
%td= inscription[key]

View file

@ -0,0 +1,24 @@
- if champ.blank?
%p= t('.not_filled')
- elsif champ.data.blank?
%p= t('.fetching_data',
identifiant: champ.identifiant)
- else
- if profile == 'usager'
- sources = champ.procedure.api_particulier_sources['pole_emploi'].keys
- i18n_sources = sources.map { |s| I18n.t("#{s}.libelle", scope: 'api_particulier.providers.pole_emploi.scopes') }
%p= t('.data_fetched', sources: i18n_sources.to_sentence, identifiant: champ.identifiant)
- if profile == 'instructeur'
%p= t('.data_fetched_title')
- champ.data.slice('identite', 'adresse', 'contact', 'inscription').keys.each do |scope|
- case scope
- when 'identite'
= render partial: 'shared/champs/pole_emploi/identite', locals: { scope: scope, identite: champ.data[scope] }
- when 'adresse'
= render partial: 'shared/champs/pole_emploi/adresse', locals: { scope: scope, adresse: champ.data[scope] }
- when 'contact'
= render partial: 'shared/champs/pole_emploi/contact', locals: { scope: scope, contact: champ.data[scope] }
- when 'inscription'
= render partial: 'shared/champs/pole_emploi/inscription', locals: { scope: scope, inscription: champ.data[scope] }

View file

@ -40,6 +40,8 @@
= render partial: "shared/champs/cnaf/show", locals: { champ: c, profile: profile }
- when TypeDeChamp.type_champs.fetch(:dgfip)
= render partial: "shared/champs/dgfip/show", locals: { champ: c, profile: profile }
- when TypeDeChamp.type_champs.fetch(:pole_emploi)
= render partial: "shared/champs/pole_emploi/show", locals: { champ: c, profile: profile }
- when TypeDeChamp.type_champs.fetch(:address)
= render partial: "shared/champs/address/show", locals: { champ: c }
- when TypeDeChamp.type_champs.fetch(:communes)

View file

@ -0,0 +1,7 @@
.pole_emploi-inputs
%div
= form.label :identifiant, t('.identifiant_label')
%p.notice= t('.identifiant_notice')
= form.text_field :identifiant,
required: champ.mandatory?,
aria: { describedby: describedby_id(champ) }

View file

@ -68,3 +68,39 @@ en:
libelle: supplements
erreurCorrectif: error correction
situationPartielle: partial status
pole_emploi:
libelle: Pôle emploi
scopes:
identite:
libelle: Identity
identifiant: identifier
civilite: civility
nom: name
nomUsage: common name
prenom: first name
sexe: gender
M: male
F: female
dateNaissance: date of birth
adresse:
libelle: Address
INSEECommune: INSEE code of the commune
codePostal: postcode
localite: city
ligneVoie: route
ligneComplementDestinataire: recipient
ligneComplementAdresse: address
ligneComplementDistribution: distribution
ligneNom: name
contact:
libelle: Contact
email: email
telephone: phone number
telephone2: phone number 2
inscription:
libelle: Registration
dateInscription: registration date
dateCessationInscription: date of deregistration
codeCertificationCNAV: CNAV certification code
codeCategorieInscription: registration category code
libelleCategorieInscription: registration category label

View file

@ -68,3 +68,39 @@ fr:
libelle: compléments
erreurCorrectif: erreur correctif
situationPartielle: situation partielle
pole_emploi:
libelle: Pôle emploi
scopes:
identite:
libelle: Identité
identifiant: identifiant
civilite: civilité
nom: nom
nomUsage: nom dusage
prenom: prénom
sexe: sexe
M: masculin
F: féminin
dateNaissance: date de naissance
adresse:
libelle: Adresse
INSEECommune: code INSEE de la commune
codePostal: code postal
localite: localité
ligneVoie: voie
ligneComplementDestinataire: destinataire
ligneComplementAdresse: adresse
ligneComplementDistribution: distribution
ligneNom: nom
contact:
libelle: Contact
email: email
telephone: téléphone
telephone2: téléphone 2
inscription:
libelle: Inscription
dateInscription: date dinscription
dateCessationInscription: date de cessation dinscription
codeCertificationCNAV: code de certification CNAV
codeCategorieInscription: code de catégorie dinscription
libelleCategorieInscription: libellé de catégorie dinscription

View file

@ -389,7 +389,7 @@ en:
message_send: "Your message has been sent to the instructor in charge of your file."
ask_deletion:
undergoingreview: "Your file is undergoing review. It is no longer possible to delete your file. To cancel the undergoingreview contact the adminitration via the mailbox."
deleted_dossier: "Your file has been successfully deleted"
soft_deleted_dossier: "Your file has been successfully deleted from your interface"
update_brouillon:
draft_saved: "Your draft has been saved."
etablissement:

View file

@ -251,7 +251,7 @@ fr:
super_admin:
<< : *default_attributes
procedure:
zone: Organisme qui met en œuvre la démarche
zone: Zone qui met en œuvre la démarche
errors:
messages:
not_a_phone: 'Numéro de téléphone invalide'
@ -402,7 +402,7 @@ fr:
message_send: "Votre message a bien été envoyé à linstructeur en charge de votre dossier."
ask_deletion:
undergoingreview: "Linstruction de votre dossier a commencé, il nest plus possible de supprimer votre dossier. Si vous souhaitez annuler linstruction contactez votre administration par la messagerie de votre dossier."
deleted_dossier: "Votre dossier a bien été supprimé."
soft_deleted_dossier: "Votre dossier a bien été supprimé de votre interface"
update_brouillon:
draft_saved: "Votre brouillon a bien été sauvegardé."
etablissement:
@ -420,7 +420,7 @@ fr:
jeton_particulier:
show:
configure_token: "Configurer le jeton API Particulier"
api_particulier_description_html: "%{app_name} utilise <a href=\"https://api.gouv.fr/les-api/api-particulier\">API Particulier</a> qui permet de récupérer les données fiscales (DGFiP) et familiales (CAF).<br />Renseignez ici le <a href=\"https://api.gouv.fr/les-api/api-particulier/demande-acces\">jeton API Particulier</a> propre à votre démarche."
api_particulier_description_html: "%{app_name} utilise <a href=\"https://api.gouv.fr/les-api/api-particulier\">API Particulier</a> qui permet d'accéder aux données familiales (CAF), aux données fiscales (DGFiP) et au statut pôle-emploi d'un citoyen.<br />Renseignez ici le <a href=\"https://api.gouv.fr/les-api/api-particulier/demande-acces\">jeton API Particulier</a> propre à votre démarche."
token_description: "Il doit contenir au minimum 15 caractères."
update:
token_ok: "Le jeton a bien été mis à jour"
@ -434,7 +434,7 @@ fr:
show:
title: "Définir les sources de données"
data_sources: "Sources de données"
explication_html: "<p>API Particulier facilite laccès des administrations aux données fiscales (DGFiP) et familiales (CAF) pour simplifier les démarches administratives mises en œuvre par les collectivités et les administrations.<br> Cela permet aux administrations daccéder à des informations certifiées à la source et ainsi : </p> <ul> <li>de saffranchir des pièces justificatives lors des démarches en ligne,</li> <li>de réduire le nombre derreurs de saisie,</li> <li>décarter le risque de fraude documentaire.</li> </ul> <p> <strong>Important&nbsp;:</strong> les disposition de l'article <a href='https://www.legifrance.gouv.fr/affichCodeArticle.do?cidTexte=LEGITEXT000031366350&amp;idArticle=LEGIARTI000031367412&amp;dateTexte=&amp;categorieLien=cid'>L144-8</a> nautorisent que léchange des informations strictement nécessaires pour traiter une démarche.<br /><br />En conséquence, ne sélectionnez ici que les données auxquelles vous aurez accès dun point de vue légal.</p>"
explication_html: "<p>API Particulier facilite laccès des administrations aux données familiales (CAF), aux données fiscales (DGFiP) et au statut pôle-emploi d'un citoyen pour simplifier les démarches administratives mises en œuvre par les collectivités et les administrations.<br> Cela permet aux administrations daccéder à des informations certifiées à la source et ainsi : </p> <ul> <li>de saffranchir des pièces justificatives lors des démarches en ligne,</li> <li>de réduire le nombre derreurs de saisie,</li> <li>décarter le risque de fraude documentaire.</li> </ul> <p> <strong>Important&nbsp;:</strong> les disposition de l'article <a href='https://www.legifrance.gouv.fr/affichCodeArticle.do?cidTexte=LEGITEXT000031366350&amp;idArticle=LEGIARTI000031367412&amp;dateTexte=&amp;categorieLien=cid'>L144-8</a> nautorisent que léchange des informations strictement nécessaires pour traiter une démarche.<br /><br />En conséquence, ne sélectionnez ici que les données auxquelles vous aurez accès dun point de vue légal.</p>"
update:
sources_ok: 'Mise à jour effectuée'
procedures:

View file

@ -38,3 +38,4 @@ fr:
annuaire_education: 'Annuaire de léducation'
cnaf: 'Données de la Caisse nationale des allocations familiales'
dgfip: 'Données de la Direction générale des Finances publiques'
pole_emploi: 'Situation Pôle emploi'

View file

@ -12,6 +12,9 @@ en:
numero_fiscal_notice: It is usually composed of 13 to 14 characters.
reference_avis_label: Tax notice reference
reference_avis_notice: It is usually composed of 13 to 14 characters.
pole_emploi:
identifiant_label: Identifier
identifiant_notice: It is usually composed of alphanumeric characters.
header:
expires_at:
brouillon: "Expires at %{date} (%{duree_conservation_totale} months after the creation of this file)"
@ -33,3 +36,9 @@ en:
fetching_data: "Fetching data for declarant No. %{numero_fiscal} with tax notice reference %{reference_avis}."
data_fetched: "Data concerning %{sources} linked to the declarant Nº %{numero_fiscal} with tax notice reference %{reference_avis} has been received from the DGFiP."
data_fetched_title: "Data received from la Direction générale des Finances publiques"
pole_emploi:
show:
not_filled: not filled
fetching_data: "Fetching data for identifier %{identifiant}."
data_fetched: "Data concerning %{sources} linked to the identifier %{identifiant} has been received from Pôle emploi."
data_fetched_title: "Data received from Pôle emploi"

View file

@ -12,6 +12,9 @@ fr:
numero_fiscal_notice: Il est généralement composé de 13 ou 14 caractères.
reference_avis_label: La référence d'avis d'imposition
reference_avis_notice: Elle est généralement composée de 13 ou 14 caractères.
pole_emploi:
identifiant_label: Identifiant
identifiant_notice: Il est généralement composé de caractères alphanumériques.
header:
expires_at:
brouillon: "Expirera le %{date} (%{duree_conservation_totale} mois après la création du dossier)"
@ -35,3 +38,9 @@ fr:
fetching_data: "La récupération automatique des données pour le déclarant Nº %{numero_fiscal} avec la référence d'avis %{reference_avis} est en cours."
data_fetched: "Des données concernant %{sources} liées au déclarant Nº %{numero_fiscal} avec la référence d'avis %{reference_avis} ont été reçues depuis la DGFiP."
data_fetched_title: "Données obtenues de la Direction générale des Finances publiques"
pole_emploi:
show:
not_filled: non renseigné
fetching_data: "La récupération automatique des données pour l'identifiant %{identifiant} est en cours."
data_fetched: "Des données concernant %{sources} liées à l'identifiant %{identifiant} ont été reçues depuis Pôle emploi."
data_fetched_title: "Données obtenues de Pôle emploi"

View file

@ -75,7 +75,11 @@ describe Administrateurs::JetonParticulierController, type: :controller do
'dgfip_revenu_fiscal_reference',
'dgfip_revenu_imposable',
'dgfip_situation_familiale',
'dgfip_situation_partielle'
'dgfip_situation_partielle',
'pole_emploi_identite',
'pole_emploi_adresse',
'pole_emploi_contact',
'pole_emploi_inscription'
)
expect(procedure.api_particulier_sources).to be_empty
end

View file

@ -193,6 +193,10 @@ FactoryBot.define do
type_de_champ { association :type_de_champ_dgfip, procedure: dossier.procedure }
end
factory :champ_pole_emploi, class: 'Champs::PoleEmploiChamp' do
type_de_champ { association :type_de_champ_pole_emploi, procedure: dossier.procedure }
end
factory :champ_siret, class: 'Champs::SiretChamp' do
type_de_champ { association :type_de_champ_siret, procedure: dossier.procedure }
association :etablissement, factory: [:etablissement]

View file

@ -206,6 +206,12 @@ FactoryBot.define do
end
end
trait :with_pole_emploi do
after(:build) do |procedure, _evaluator|
build(:type_de_champ_pole_emploi, procedure: procedure)
end
end
trait :with_explication do
after(:build) do |procedure, _evaluator|
build(:type_de_champ_explication, procedure: procedure)

View file

@ -160,6 +160,9 @@ FactoryBot.define do
factory :type_de_champ_dgfip do
type_champ { TypeDeChamp.type_champs.fetch(:dgfip) }
end
factory :type_de_champ_pole_emploi do
type_champ { TypeDeChamp.type_champs.fetch(:pole_emploi) }
end
factory :type_de_champ_carte do
type_champ { TypeDeChamp.type_champs.fetch(:carte) }
end

View file

@ -25,7 +25,7 @@ http_interactions:
Content-Type:
- application/json
Content-Length:
- '228'
- '257'
Connection:
- keep-alive
Keep-Alive:
@ -38,6 +38,6 @@ http_interactions:
- max-age=15552000
body:
encoding: UTF-8
string: '{"_id":"1d99db5a-a099-4314-ad2f-2707c6b505a6","name":"Application de sandbox","scopes":["cnaf_allocataires","cnaf_enfants","cnaf_adresse","cnaf_quotient_familial","dgfip_declarant1_nom","dgfip_declarant1_nom_naissance","dgfip_declarant1_prenoms","dgfip_declarant1_date_naissance","dgfip_declarant2_nom","dgfip_declarant2_nom_naissance","dgfip_declarant2_prenoms","dgfip_declarant2_date_naissance","dgfip_date_recouvrement","dgfip_date_etablissement","dgfip_adresse_fiscale_taxation","dgfip_adresse_fiscale_annee","dgfip_nombre_parts","dgfip_nombre_personnes_a_charge","dgfip_situation_familiale","dgfip_revenu_brut_global","dgfip_revenu_imposable","dgfip_impot_revenu_net_avant_corrections","dgfip_montant_impot","dgfip_revenu_fiscal_reference","dgfip_annee_impot","dgfip_annee_revenus","dgfip_erreur_correctif","dgfip_situation_partielle"]}'
string: '{"_id":"1d99db5a-a099-4314-ad2f-2707c6b505a6","name":"Application de sandbox","scopes":["cnaf_allocataires","cnaf_enfants","cnaf_adresse","cnaf_quotient_familial","dgfip_declarant1_nom","dgfip_declarant1_nom_naissance","dgfip_declarant1_prenoms","dgfip_declarant1_date_naissance","dgfip_declarant2_nom","dgfip_declarant2_nom_naissance","dgfip_declarant2_prenoms","dgfip_declarant2_date_naissance","dgfip_date_recouvrement","dgfip_date_etablissement","dgfip_adresse_fiscale_taxation","dgfip_adresse_fiscale_annee","dgfip_nombre_parts","dgfip_nombre_personnes_a_charge","dgfip_situation_familiale","dgfip_revenu_brut_global","dgfip_revenu_imposable","dgfip_impot_revenu_net_avant_corrections","dgfip_montant_impot","dgfip_revenu_fiscal_reference","dgfip_annee_impot","dgfip_annee_revenus","dgfip_erreur_correctif","dgfip_situation_partielle", "pole_emploi_identite","pole_emploi_adresse","pole_emploi_contact","pole_emploi_inscription"]}'
recorded_at: Tue, 16 Mar 2021 15:25:24 GMT
recorded_with: VCR 6.0.0

View file

@ -0,0 +1,76 @@
---
http_interactions:
- request:
method: get
uri: https://particulier.api.gouv.fr/api/v2/situations-pole-emploi?identifiant=georges_moustaki_77
body:
encoding: US-ASCII
string: ''
headers:
User-Agent:
- demarches-simplifiees.fr
Accept:
- application/json
X-Api-Key:
- 06fd8675601267d2988cbbdef56ecb0de1d45223
Expect:
- ''
response:
status:
code: 200
message: OK
headers:
Date:
- Tue, 16 Mar 2021 17:01:19 GMT
Content-Type:
- application/json; charset=utf-8
Content-Length:
- '651'
Connection:
- keep-alive
Keep-Alive:
- timeout=5
Access-Control-Allow-Credentials:
- 'true'
Access-Control-Expose-Headers:
- Range,Content-Range,X-Content-Range,X-API-Key
Etag:
- W/"2eb-A0NiRd+gbJKIAT0y4tR4j9tjXb0"
Server:
- nginx
Strict-Transport-Security:
- max-age=15552000
- max-age=15552000
Vary:
- Origin, Accept
X-Gravitee-Request-Id:
- 7bfb7f99-ac2d-4443-bb7f-99ac2d0443c5
X-Gravitee-Transaction-Id:
- f5dca8b3-2ab7-4c9a-9ca8-b32ab70c9a2b
body:
encoding: ASCII-8BIT
string: '{
"identifiant": "georges_moustaki_77",
"civilite": "M.",
"nom": "Moustaki",
"nomUsage": "Moustaki",
"prenom": "Georges",
"sexe": "M",
"dateNaissance": "1934-05-03",
"adresse": {
"INSEECommune": "75118",
"codePostal": "75018",
"localite": "75018 Paris",
"ligneVoie": "3 rue des Huttes",
"ligneNom": "MOUSTAKI"
},
"email": "georges@moustaki.fr",
"telephone": "0629212921",
"dateInscription": "1965-05-03",
"dateCessationInscription": "1966-05-03",
"codeCertificationCNAV": "VC",
"codeCategorieInscription": 1,
"libelleCategorieInscription": "PERSONNE SANS EMPLOI DISPONIBLE DUREE INDETERMINEE PLEIN TPS"
}'
recorded_at: Tue, 16 Mar 2021 17:01:18 GMT
recorded_with: VCR 6.0.0

View file

@ -0,0 +1,76 @@
---
http_interactions:
- request:
method: get
uri: https://particulier.api.gouv.fr/api/v2/situations-pole-emploi?identifiant=georges_moustaki_77
body:
encoding: US-ASCII
string: ''
headers:
User-Agent:
- demarches-simplifiees.fr
Accept:
- application/json
X-Api-Key:
- 06fd8675601267d2988cbbdef56ecb0de1d45223
Expect:
- ''
response:
status:
code: 200
message: OK
headers:
Date:
- Tue, 16 Mar 2021 17:01:19 GMT
Content-Type:
- application/json; charset=utf-8
Content-Length:
- '747'
Connection:
- keep-alive
Keep-Alive:
- timeout=5
Access-Control-Allow-Credentials:
- '651'
Access-Control-Expose-Headers:
- Range,Content-Range,X-Content-Range,X-API-Key
Etag:
- W/"2eb-A0NiRd+gbJKIAT0y4tR4j9tjXb0"
Server:
- nginx
Strict-Transport-Security:
- max-age=15552000
- max-age=15552000
Vary:
- Origin, Accept
X-Gravitee-Request-Id:
- 7bfb7f99-ac2d-4443-bb7f-99ac2d0443c5
X-Gravitee-Transaction-Id:
- f5dca8b3-2ab7-4c9a-9ca8-b32ab70c9a2b
body:
encoding: ASCII-8BIT
string: '{
"identifiant": "georges_moustaki_77",
"civilite": "M.",
"nom": "Moustaki",
"nomUsage": "Moustaki",
"prenom": "Georges",
"sexe": "X",
"dateNaissance": "1934-05-03",
"adresse": {
"INSEECommune": "75118",
"codePostal": "75018",
"localite": "75018 Paris",
"ligneVoie": "3 rue des Huttes",
"ligneNom": "MOUSTAKI"
},
"email": "georges@moustaki.fr",
"telephone": "0629212921",
"dateInscription": "1965-05-03",
"dateCessationInscription": "1966-05-03",
"codeCertificationCNAV": "VC",
"codeCategorieInscription": 1,
"libelleCategorieInscription": "PERSONNE SANS EMPLOI DISPONIBLE DUREE INDETERMINEE PLEIN TPS"
}'
recorded_at: Tue, 16 Mar 2021 17:01:18 GMT
recorded_with: VCR 6.0.0

View file

@ -0,0 +1,29 @@
{
"identite": {
"identifiant": "georges_moustaki_77",
"civilite": "M.",
"nom": "Moustaki",
"nomUsage": "Moustaki",
"prenom": "Georges",
"sexe": "M",
"dateNaissance": "1934-05-03"
},
"adresse": {
"INSEECommune": "75118",
"codePostal": "75018",
"localite": "75018 Paris",
"ligneVoie": "3 rue des Huttes",
"ligneNom": "MOUSTAKI"
},
"contact": {
"email": "georges@moustaki.fr",
"telephone": "0629212921"
},
"inscription": {
"dateInscription": "1965-05-03",
"dateCessationInscription": "1966-05-03",
"codeCertificationCNAV": "VC",
"codeCategorieInscription": 1,
"libelleCategorieInscription": "PERSONNE SANS EMPLOI DISPONIBLE DUREE INDETERMINEE PLEIN TPS"
}
}

View file

@ -0,0 +1,45 @@
RSpec.describe Types::DossierType, type: :graphql do
let(:query) { DOSSIER_QUERY }
let(:context) { { internal_use: true } }
let(:variables) { {} }
subject { API::V2::Schema.execute(query, variables: variables, context: context) }
let(:data) { subject['data'].deep_symbolize_keys }
let(:errors) { subject['errors'].deep_symbolize_keys }
describe 'dossier with attestation' do
let(:dossier) { create(:dossier, :accepte, :with_attestation) }
let(:query) { DOSSIER_WITH_ATTESTATION_QUERY }
let(:variables) { { number: dossier.id } }
it { expect(data[:dossier][:attestation]).not_to be_nil }
context 'when attestation is nil' do
before do
dossier.update(attestation: nil)
end
it { expect(data[:dossier][:attestation]).to be_nil }
end
end
DOSSIER_QUERY = <<-GRAPHQL
query($number: Int!) {
dossier(number: $number) {
id
number
}
}
GRAPHQL
DOSSIER_WITH_ATTESTATION_QUERY = <<-GRAPHQL
query($number: Int!) {
dossier(number: $number) {
attestation {
url
}
}
}
GRAPHQL
end

View file

@ -37,7 +37,11 @@ describe APIParticulier::API do
'dgfip_revenu_fiscal_reference',
'dgfip_revenu_imposable',
'dgfip_situation_familiale',
'dgfip_situation_partielle'
'dgfip_situation_partielle',
'pole_emploi_identite',
'pole_emploi_adresse',
'pole_emploi_contact',
'pole_emploi_inscription'
)
end
end

View file

@ -0,0 +1,63 @@
describe APIParticulier::PoleEmploiAdapter do
let(:adapter) { described_class.new(api_particulier_token, identifiant, requested_sources) }
before { stub_const('API_PARTICULIER_URL', 'https://particulier.api.gouv.fr/api') }
describe '#to_params' do
let(:api_particulier_token) { '06fd8675601267d2988cbbdef56ecb0de1d45223' }
let(:identifiant) { 'georges_moustaki_77' }
subject { VCR.use_cassette(cassette) { adapter.to_params } }
context 'when the api answer is valid' do
let(:cassette) { 'api_particulier/success/situation_pole_emploi' }
context 'when the token has all the pole emploi scopes' do
context 'and all the sources are requested' do
let(:requested_sources) do
{
'pole_emploi' => {
'identite' => ['identifiant', 'civilite', 'nom', 'nomUsage', 'prenom', 'sexe', 'dateNaissance'],
'adresse' => ['INSEECommune', 'codePostal', 'localite', 'ligneVoie', 'ligneComplementDestinataire', 'ligneComplementAdresse', 'ligneComplementDistribution', 'ligneNom'],
'contact' => ['email', 'telephone', 'telephone2'],
'inscription' => ['dateInscription', 'dateCessationInscription', 'codeCertificationCNAV', 'codeCategorieInscription', 'libelleCategorieInscription']
}
}
end
let(:result) { JSON.parse(File.read('spec/fixtures/files/api_particulier/situation_pole_emploi.json')) }
it { is_expected.to eq(result) }
end
context 'when no sources is requested' do
let(:requested_sources) { {} }
it { is_expected.to eq({}) }
end
context 'when an address name is requested' do
let(:requested_sources) { { 'pole_emploi' => { 'adresse' => ['ligneNom'] } } }
it { is_expected.to eq('adresse' => { 'ligneNom' => 'MOUSTAKI' }) }
end
context 'when a first name is requested' do
let(:requested_sources) { { 'pole_emploi' => { 'identite' => ['prenom'] } } }
it { is_expected.to eq('identite' => { 'prenom' => 'Georges' }) }
end
end
end
context 'when the api answer is invalid' do
let(:cassette) { 'api_particulier/success/situation_pole_emploi_invalid' }
context 'when no sources is requested' do
let(:requested_sources) { {} }
it { expect { subject }.to raise_error(APIParticulier::PoleEmploiAdapter::InvalidSchemaError) }
end
end
end
end

View file

@ -47,6 +47,21 @@ describe APIParticulier::Services::SourcesService do
it { is_expected.to match(dgfip_avis_imposition_et_adresse) }
end
context 'when a procedure has a pole_emploi_identite and a pole_emploi_adresse scopes' do
let(:api_particulier_scopes) { ['pole_emploi_identite', 'pole_emploi_adresse'] }
let(:pole_emploi_identite_et_adresse) do
{
'pole_emploi' => {
'identite' => ['identifiant', 'civilite', 'nom', 'nomUsage', 'prenom', 'sexe', 'dateNaissance'],
'adresse' => ['INSEECommune', 'codePostal', 'localite', 'ligneVoie', 'ligneComplementDestinataire', 'ligneComplementAdresse', 'ligneComplementDistribution', 'ligneNom']
}
}
end
it { is_expected.to match(pole_emploi_identite_et_adresse) }
end
context 'when a procedure has an unknown scope' do
let(:api_particulier_scopes) { ['unknown_scope'] }

View file

@ -59,7 +59,7 @@ describe Helpscout::FormAdapter do
expect(api).to have_received(:create_conversation)
.with(email, subject, text, nil)
expect(api).to have_received(:add_tags)
.with(conversation_id, tags)
.with(conversation_id, tags + ['contact form'])
end
end

View file

@ -0,0 +1,56 @@
describe Champs::PoleEmploiChamp, type: :model do
let(:champ) { described_class.new }
describe 'identifiant' do
before do
champ.identifiant = 'georges_moustaki_77'
end
it 'saves identifiant' do
expect(champ.identifiant).to eq('georges_moustaki_77')
end
end
describe 'external_id' do
context 'when no data is given' do
before do
champ.identifiant = ''
champ.save
end
it { expect(champ.external_id).to be_nil }
end
context 'when all data required for an external fetch are given' do
before do
champ.identifiant = 'georges_moustaki_77'
champ.save
end
it { expect(JSON.parse(champ.external_id)).to eq("identifiant" => "georges_moustaki_77") }
end
end
describe '#validate' do
let(:champ) { described_class.new(dossier: create(:dossier), type_de_champ: create(:type_de_champ_pole_emploi)) }
let(:validation_context) { :create }
subject { champ.valid?(validation_context) }
before do
champ.identifiant = identifiant
end
context 'when identifiant is valid' do
let(:identifiant) { 'georges_moustaki_77' }
it { is_expected.to be true }
end
context 'when identifiant is nil' do
let(:identifiant) { nil }
it { is_expected.to be true }
end
end
end

View file

@ -79,6 +79,7 @@ describe ProcedureExportService do
"annuaire_education",
"cnaf",
"dgfip",
"pole_emploi",
"text"
]
end
@ -168,6 +169,7 @@ describe ProcedureExportService do
"annuaire_education",
"cnaf",
"dgfip",
"pole_emploi",
"text"
]
end
@ -253,6 +255,7 @@ describe ProcedureExportService do
"annuaire_education",
"cnaf",
"dgfip",
"pole_emploi",
"text"
]
end

View file

@ -20,6 +20,12 @@ describe 'fetch API Particulier Data', js: true do
'foyer_fiscal' => ['adresse', 'annee', 'nombreParts', 'nombrePersonnesCharge', 'situationFamille'],
'agregats_fiscaux' => ['anneeImpots', 'anneeRevenus', 'impotRevenuNetAvantCorrections', 'montantImpot', 'revenuBrutGlobal', 'revenuFiscalReference', 'revenuImposable'],
'complements' => ['situationPartielle', 'erreurCorrectif']
},
'pole_emploi' => {
'identite' => ['identifiant', 'civilite', 'nom', 'nomUsage', 'prenom', 'sexe', 'dateNaissance'],
'adresse' => ['INSEECommune', 'codePostal', 'localite', 'ligneVoie', 'ligneComplementDestinataire', 'ligneComplementAdresse', 'ligneComplementDistribution', 'ligneNom'],
'contact' => ['email', 'telephone', 'telephone2'],
'inscription' => ['dateInscription', 'dateCessationInscription', 'codeCertificationCNAV', 'codeCategorieInscription', 'libelleCategorieInscription']
}
}
end
@ -29,7 +35,7 @@ describe 'fetch API Particulier Data', js: true do
Flipper.enable(:api_particulier)
end
context 'when an administrateur is logged' do
context 'when an administrateur is logged in' do
let(:procedure) do
create(:procedure, :with_service, :with_instructeur,
aasm_state: :brouillon,
@ -56,14 +62,14 @@ describe 'fetch API Particulier Data', js: true do
expect(page).to have_current_path(admin_procedure_api_particulier_sources_path(procedure))
['allocataires', 'enfants'].each do |scope|
within("##{scope}") do
within("#cnaf-#{scope}") do
check('noms et prénoms')
check('date de naissance')
check('sexe')
end
end
within('#adresse') do
within('#cnaf-adresse') do
check('identité')
check('complément didentité')
check('complément didentité géographique')
@ -73,14 +79,14 @@ describe 'fetch API Particulier Data', js: true do
check('pays')
end
within('#quotient_familial') do
within('#cnaf-quotient_familial') do
check('quotient familial')
check('année')
check('mois')
end
['declarant1', 'declarant2'].each do |scope|
within("##{scope}") do
within("#dgfip-#{scope}") do
check('nom')
check('nom de naissance')
check('prénoms')
@ -88,13 +94,13 @@ describe 'fetch API Particulier Data', js: true do
end
end
scroll_to(find('#echeance_avis'))
within ('#echeance_avis') do
scroll_to(find('#dgfip-echeance_avis'))
within ('#dgfip-echeance_avis') do
check('date de recouvrement')
check("date détablissement")
end
within('#foyer_fiscal') do
within('#dgfip-foyer_fiscal') do
check('année')
check('adresse')
check('nombre de parts')
@ -102,7 +108,7 @@ describe 'fetch API Particulier Data', js: true do
check('nombre de personnes à charge')
end
within('#agregats_fiscaux') do
within('#dgfip-agregats_fiscaux') do
check('revenu brut global')
check('revenu imposable')
check('impôt sur le revenu net avant correction')
@ -112,22 +118,58 @@ describe 'fetch API Particulier Data', js: true do
check('année des revenus')
end
within('#complements') do
within('#dgfip-complements') do
check('erreur correctif')
check('situation partielle')
end
within('#pole_emploi-identite') do
check('identifiant')
check('civilité')
check('nom')
check("nom dusage")
check('prénom')
check('sexe')
check('date de naissance')
end
within('#pole_emploi-adresse') do
check('code INSEE de la commune')
check('code postal')
check('localité')
check('voie')
check('destinataire')
check('adresse')
check('distribution')
check('nom')
end
within('#pole_emploi-contact') do
check('email')
check('téléphone')
check('téléphone 2')
end
within('#pole_emploi-inscription') do
check("date dinscription")
check("date de cessation dinscription")
check('code de certification CNAV')
check("code de catégorie dinscription")
check("libellé de catégorie dinscription")
end
click_on 'Enregistrer'
within('#enfants') do
within('#cnaf-enfants') do
expect(find('input[value=nomPrenom]')).to be_checked
end
procedure.reload
expect(procedure.api_particulier_sources.keys).to contain_exactly('cnaf', 'dgfip')
expect(procedure.api_particulier_sources.keys).to contain_exactly('cnaf', 'dgfip', 'pole_emploi')
expect(procedure.api_particulier_sources['cnaf'].keys).to contain_exactly('adresse', 'allocataires', 'enfants', 'quotient_familial')
expect(procedure.api_particulier_sources['dgfip'].keys).to contain_exactly('declarant1', 'declarant2', 'echeance_avis', 'foyer_fiscal', 'agregats_fiscaux', 'complements')
expect(procedure.api_particulier_sources['pole_emploi'].keys).to contain_exactly('identite', 'adresse', 'contact', 'inscription')
procedure.api_particulier_sources.each do |provider, scopes|
scopes.each do |scope, fields|
@ -153,17 +195,18 @@ describe 'fetch API Particulier Data', js: true do
end
end
context 'when an user is logged' do
context 'when a user is logged in' do
let(:user) { create(:user) }
let(:api_particulier_token) { '29eb50b65f64e8e00c0847a8bbcbd150e1f847' }
let(:numero_allocataire) { '5843972' }
let(:code_postal) { '92110' }
let(:numero_fiscal) { '2097699999077' }
let(:reference_avis) { '2097699999077' }
let(:instructeur) { create(:instructeur) }
let(:identifiant) { 'georges_moustaki_77' }
let(:api_particulier_token) { '29eb50b65f64e8e00c0847a8bbcbd150e1f847' }
let(:procedure) do
create(:procedure, :for_individual, :with_service, :with_cnaf, :with_dgfip, :published,
create(:procedure, :for_individual, :with_service, :with_cnaf, :with_dgfip, :with_pole_emploi, :published,
libelle: 'libellé de la procédure',
path: 'libelle-de-la-procedure',
instructeurs: [instructeur],
@ -173,54 +216,124 @@ describe 'fetch API Particulier Data', js: true do
before { login_as user, scope: :user }
scenario 'it can fill an cnaf champ' do
visit commencer_path(path: procedure.path)
click_on 'Commencer la démarche'
context 'CNAF' do
scenario 'it can fill an cnaf champ' do
visit commencer_path(path: procedure.path)
click_on 'Commencer la démarche'
choose 'Monsieur'
fill_in 'individual_nom', with: 'Nom'
fill_in 'individual_prenom', with: 'Prenom'
choose 'Monsieur'
fill_in 'individual_nom', with: 'Nom'
fill_in 'individual_prenom', with: 'Prenom'
click_button('Continuer')
click_button('Continuer')
fill_in 'Le numéro dallocataire CAF', with: numero_allocataire
fill_in 'Le code postal', with: 'wrong_code'
fill_in 'Le numéro dallocataire CAF', with: numero_allocataire
fill_in 'Le code postal', with: 'wrong_code'
blur
expect(page).to have_css('span', text: 'Brouillon enregistré', visible: true)
blur
expect(page).to have_css('span', text: 'Brouillon enregistré', visible: true)
dossier = Dossier.last
expect(dossier.champs.first.code_postal).to eq('wrong_code')
dossier = Dossier.last
expect(dossier.champs.first.code_postal).to eq('wrong_code')
click_on 'Déposer le dossier'
expect(page).to have_content(/code postal doit posséder 5 caractères/)
click_on 'Déposer le dossier'
expect(page).to have_content(/code postal doit posséder 5 caractères/)
fill_in 'Le code postal', with: code_postal
fill_in 'Le code postal', with: code_postal
VCR.use_cassette('api_particulier/success/composition_familiale') do
perform_enqueued_jobs { click_on 'Déposer le dossier' }
VCR.use_cassette('api_particulier/success/composition_familiale') do
perform_enqueued_jobs { click_on 'Déposer le dossier' }
end
visit demande_dossier_path(dossier)
expect(page).to have_content(/Des données.*ont été reçues depuis la CAF/)
log_out
login_as instructeur.user, scope: :user
visit instructeur_dossier_path(procedure, dossier)
expect(page).to have_content('code postal et ville 92110 Clichy')
expect(page).to have_content('identité Mr SNOW Eric')
expect(page).to have_content('complément didentité ne connait rien')
expect(page).to have_content('numéro et rue 109 rue La Boétie')
expect(page).to have_content('pays FRANCE')
expect(page).to have_content('complément didentité géographique au nord de paris')
expect(page).to have_content('lieu-dit glagla')
expect(page).to have_content('ERIC SNOW masculin 07/01/1991')
expect(page).to have_content('SANSA SNOW féminin 15/01/1992')
expect(page).to have_content('PAUL SNOW masculin 04/01/2018')
expect(page).to have_content('1856 6 2021')
end
end
visit demande_dossier_path(dossier)
expect(page).to have_content(/Des données.*ont été reçues depuis la CAF/)
context 'Pôle emploi' do
let(:api_particulier_token) { '06fd8675601267d2988cbbdef56ecb0de1d45223' }
log_out
scenario 'it can fill a Pôle emploi field' do
visit commencer_path(path: procedure.path)
click_on 'Commencer la démarche'
login_as instructeur.user, scope: :user
choose 'Monsieur'
fill_in 'individual_nom', with: 'Moustaki'
fill_in 'individual_prenom', with: 'Georges'
visit instructeur_dossier_path(procedure, dossier)
click_button('Continuer')
expect(page).to have_content('code postal et ville 92110 Clichy')
expect(page).to have_content('identité Mr SNOW Eric')
expect(page).to have_content('complément didentité ne connait rien')
expect(page).to have_content('numéro et rue 109 rue La Boétie')
expect(page).to have_content('pays FRANCE')
expect(page).to have_content('complément didentité géographique au nord de paris')
expect(page).to have_content('lieu-dit glagla')
expect(page).to have_content('ERIC SNOW masculin 07/01/1991')
expect(page).to have_content('SANSA SNOW féminin 15/01/1992')
expect(page).to have_content('PAUL SNOW masculin 04/01/2018')
expect(page).to have_content('1856 6 2021')
fill_in "Identifiant", with: 'wrong code'
blur
expect(page).to have_css('span', text: 'Brouillon enregistré', visible: true)
dossier = Dossier.last
pole_emploi_champ = dossier.champs.third
expect(pole_emploi_champ.identifiant).to eq('wrong code')
fill_in "Identifiant", with: identifiant
VCR.use_cassette('api_particulier/success/situation_pole_emploi') do
perform_enqueued_jobs { click_on 'Déposer le dossier' }
end
visit demande_dossier_path(dossier)
expect(page).to have_content(/Des données.*ont été reçues depuis Pôle emploi/)
log_out
login_as instructeur.user, scope: :user
visit instructeur_dossier_path(procedure, dossier)
expect(page).to have_content('identifiant georges_moustaki_77')
expect(page).to have_content('civilité M.')
expect(page).to have_content('nom Moustaki')
expect(page).to have_content("nom dusage Moustaki")
expect(page).to have_content('prénom Georges')
expect(page).to have_content('sexe masculin')
expect(page).to have_content('date de naissance 3 mai 1934')
expect(page).to have_content('code INSEE de la commune 75118')
expect(page).to have_content('code postal 75018')
expect(page).to have_content('localité 75018 Paris')
expect(page).to have_content('voie 3 rue des Huttes')
expect(page).to have_content('nom MOUSTAKI')
expect(page).to have_content('email georges@moustaki.fr')
expect(page).to have_content('téléphone 0629212921')
expect(page).to have_content("date dinscription 3 mai 1965")
expect(page).to have_content("date de cessation dinscription 3 mai 1966")
expect(page).to have_content('code de certification CNAV VC')
expect(page).to have_content("code de catégorie dinscription 1")
expect(page).to have_content("libellé de catégorie dinscription PERSONNE SANS EMPLOI DISPONIBLE DUREE INDETERMINEE PLEIN TPS")
expect(page).not_to have_content('téléphone 2')
expect(page).not_to have_content('destinataire')
expect(page).not_to have_content('adresse')
expect(page).not_to have_content('distribution')
end
end
scenario 'it can fill a DGFiP field' do

View file

@ -76,7 +76,7 @@ describe 'user access to the list of their dossiers' do
end
end
expect(page).to have_content('Votre dossier a bien été supprimé.')
expect(page).to have_content('Votre dossier a bien été supprimé')
expect(page).not_to have_content(dossier_brouillon.procedure.libelle)
end
end