describe 'The user' do let(:password) { 'my-s3cure-p4ssword' } let!(:user) { create(:user, password: password) } let!(:procedure) { create(:procedure, :published, :for_individual, :with_all_champs_mandatory) } let(:user_dossier) { user.dossiers.first } let!(:dossier_to_link) { create(:dossier) } scenario 'fill a dossier', js: true do log_in(user, procedure) fill_individual # fill data fill_in('text *', with: 'super texte') fill_in('textarea', with: 'super textarea') fill_in('date', with: '12-12-2012') select_date_and_time(Time.zone.parse('06/01/2030 7h05'), form_id_for_datetime('datetime')) fill_in('number', with: '42') fill_in('decimal_number', with: '17') fill_in('integer_number', with: '12') scroll_to(find_field('checkbox'), align: :center) check('checkbox') choose('Madame') fill_in('email', with: 'loulou@yopmail.com') fill_in('phone', with: '0123456789') scroll_to(find_field('Non'), align: :center) choose('Non') choose('val2') check('val1') check('val3') select('bravo', from: form_id_for('simple_choice_drop_down_list_long')) select_combobox('multiple_choice_drop_down_list_long', 'alp', 'alpha') select_combobox('multiple_choice_drop_down_list_long', 'cha', 'charly') select_combobox('pays', 'aust', 'Australie') select_combobox('regions', 'Ma', 'Martinique') select_combobox('departements', 'Ai', '02 - Aisne') select_combobox('communes', 'Ai', '02 - Aisne', check: false) select_combobox('communes', 'Ambl', 'Ambléon (01300)') check('engagement') fill_in('dossier_link', with: dossier_to_link.id) find('.editable-champ-piece_justificative input[type=file]').attach_file(Rails.root + 'spec/fixtures/files/file.pdf') expect(page).to have_css('span', text: 'Votre brouillon est automatiquement enregistré', visible: true) blur expect(page).to have_css('span', text: 'Brouillon enregistré', visible: true) # check data on the dossier from db # Sometimes, `user_dossier.champs` are not yet all updated with the new values # when we first load `user_dossier`, causing random errors. # Strategy is to retry & reload them if necessary for a few seconds, # and raise expectation error instead of timeout error. last_expection_error = nil begin Timeout.timeout(Capybara.default_max_wait_time) do expect(user_dossier).to be_brouillon expect(champ_value_for('text')).to eq('super texte') expect(champ_value_for('textarea')).to eq('super textarea') expect(champ_value_for('date')).to eq('2012-12-12') expect(champ_value_for('datetime')).to eq('06/01/2030 07:05') expect(champ_value_for('number')).to eq('42') expect(champ_value_for('decimal_number')).to eq('17') expect(champ_value_for('integer_number')).to eq('12') expect(champ_value_for('checkbox')).to eq('on') expect(champ_value_for('civilite')).to eq('Mme') expect(champ_value_for('email')).to eq('loulou@yopmail.com') expect(champ_value_for('phone')).to eq('0123456789') expect(champ_value_for('yes_no')).to eq('false') expect(champ_value_for('simple_drop_down_list')).to eq('val2') expect(champ_value_for('simple_choice_drop_down_list_long')).to eq('bravo') expect(JSON.parse(champ_value_for('multiple_choice_drop_down_list_long'))).to match(['alpha', 'charly']) expect(JSON.parse(champ_value_for('multiple_drop_down_list'))).to match(['val1', 'val3']) expect(champ_value_for('pays')).to eq('Australie') expect(champ_value_for('regions')).to eq('Martinique') expect(champ_value_for('departements')).to eq('02 - Aisne') expect(champ_value_for('communes')).to eq('Ambléon (01300)') expect(champ_value_for('engagement')).to eq('on') expect(champ_value_for('dossier_link')).to eq(dossier_to_link.id.to_s) expect(champ_value_for('piece_justificative')).to be_nil # antivirus hasn't approved the file yet rescue RSpec::Expectations::ExpectationNotMetError => e Rails.logger.debug "Error #{e.message.tr("\n", " ")}, will retry" last_expection_error = e sleep(0.1) retry end rescue Timeout::Error => e raise last_expection_error || e end ## check data on the gui expect(page).to have_field('text', with: 'super texte') expect(page).to have_field('textarea', with: 'super textarea') expect(page).to have_field('date', with: '2012-12-12') check_date_and_time(Time.zone.parse('06/01/2030 7:05'), form_id_for_datetime('datetime')) expect(page).to have_field('number', with: '42') expect(page).to have_checked_field('checkbox') expect(page).to have_checked_field('Madame') expect(page).to have_field('email', with: 'loulou@yopmail.com') expect(page).to have_field('phone', with: '0123456789') expect(page).to have_checked_field('Non') expect(page).to have_checked_field('val2') expect(page).to have_checked_field('val1') expect(page).to have_checked_field('val3') expect(page).to have_selected_value('simple_choice_drop_down_list_long', selected: 'bravo') check_selected_value('multiple_choice_drop_down_list_long', with: ['alpha', 'charly']) check_selected_value('pays', with: 'Australie') check_selected_value('regions', with: 'Martinique') check_selected_value('departements', with: '02 - Aisne') check_selected_value('communes', with: 'Ambléon (01300)') expect(page).to have_checked_field('engagement') expect(page).to have_field('dossier_link', with: dossier_to_link.id.to_s) expect(page).to have_text('file.pdf') expect(page).to have_text('analyse antivirus en cours') end let(:procedure_with_repetition) do create(:procedure, :published, :for_individual, :with_repetition) end scenario 'fill a dossier with repetition', js: true do log_in(user, procedure_with_repetition) fill_individual fill_in('sub type de champ', with: 'super texte') expect(page).to have_field('sub type de champ', with: 'super texte') click_on 'Ajouter un élément pour' within '.repetition .row:first-child' do fill_in('sub type de champ', with: 'un autre texte') end expect(page).to have_content('Supprimer', count: 2) blur expect(page).to have_css('span', text: 'Brouillon enregistré', visible: true) expect(page).to have_content('Supprimer', count: 2) within '.repetition .row:first-child' do click_on 'Supprimer l’élément' end blur expect(page).to have_css('span', text: 'Brouillon enregistré', visible: true) expect(page).to have_content('Supprimer', count: 1) end let(:simple_procedure) { create(:procedure, :published, :for_individual, types_de_champ_public: [{ mandatory: true, libelle: 'texte obligatoire' }, { mandatory: false, libelle: 'texte optionnel' }]) } scenario 'save an incomplete dossier as draft but cannot not submit it', js: true do log_in(user, simple_procedure) fill_individual # Check an incomplete dossier can be saved as a draft, even when mandatory fields are missing fill_in('texte optionnel', with: 'ça ne suffira pas') blur expect(page).to have_css('span', text: 'Brouillon enregistré', visible: true) expect(page).to have_current_path(brouillon_dossier_path(user_dossier)) # Check an incomplete dossier cannot be submitted when mandatory fields are missing click_on 'Déposer le dossier' expect(user_dossier.reload.brouillon?).to be(true) expect(page).to have_current_path(brouillon_dossier_path(user_dossier)) # Check a dossier can be submitted when all mandatory fields are filled fill_in('texte obligatoire', with: 'super texte') click_on 'Déposer le dossier' expect(user_dossier.reload.en_construction?).to be(true) expect(champ_value_for('texte obligatoire')).to eq('super texte') expect(page).to have_current_path(merci_dossier_path(user_dossier)) end scenario 'extends dossier experation date more than one time, ', js: true do simple_procedure.update(procedure_expires_when_termine_enabled: true) allow(simple_procedure).to receive(:feature_enabled?).with(:procedure_process_expired_dossiers_termine).and_return(true) user_old_dossier = create(:dossier, procedure: simple_procedure, created_at: simple_procedure.duree_conservation_dossiers_dans_ds.month.ago, user: user) login_as(user, scope: :user) visit brouillon_dossier_path(user_old_dossier) expect(page).to have_css('.card-title', text: 'Votre dossier va expirer', visible: true) find('#test-user-repousser-expiration').click expect(page).to have_no_selector('#test-user-repousser-expiration') Timecop.freeze(simple_procedure.duree_conservation_dossiers_dans_ds.month.from_now) do visit brouillon_dossier_path(user_old_dossier) expect(page).to have_css('.card-title', text: 'Votre dossier va expirer', visible: true) find('#test-user-repousser-expiration').click expect(page).to have_no_selector('#test-user-repousser-expiration') end end let(:procedure_with_pj) { create(:procedure, :published, :for_individual, types_de_champ_public: [{ type: :piece_justificative, mandatory: true, libelle: 'Pièce justificative' }]) } let(:procedure_with_pjs) { create(:procedure, :published, :for_individual, types_de_champ_public: [{ type: :piece_justificative, mandatory: true, libelle: 'Pièce justificative 1' }, { type: :piece_justificative, mandatory: true, libelle: 'Pièce justificative 2' }]) } let(:old_procedure_with_disabled_pj_validation) { create(:procedure, :published, :for_individual, types_de_champ_public: [{ type: :piece_justificative, mandatory: true, libelle: 'Pièce justificative 1', skip_pj_validation: true }]) } scenario 'add an attachment', js: true do log_in(user, procedure_with_pjs) fill_individual # Add attachments find_field('Pièce justificative 1').attach_file(Rails.root + 'spec/fixtures/files/file.pdf') find_field('Pièce justificative 2').attach_file(Rails.root + 'spec/fixtures/files/RIB.pdf') # Expect the files to be uploaded immediately expect(page).to have_text('analyse antivirus en cours', count: 2) expect(page).to have_text('file.pdf') expect(page).to have_text('RIB.pdf') # Expect the submit buttons to be enabled expect(page).to have_button('Déposer le dossier', disabled: false) # Reload the current page visit current_path # Expect the files to have been saved on the dossier expect(page).to have_text('file.pdf') expect(page).to have_text('RIB.pdf') end scenario 'add an invalid attachment on an old procedure where pj validation is disabled', js: true do log_in(user, old_procedure_with_disabled_pj_validation) fill_individual # Test invalid file type attach_file('Pièce justificative 1', Rails.root + 'spec/fixtures/files/invalid_file_format.json') expect(page).to have_no_text('La pièce justificative n’est pas d’un type accepté') expect(page).to have_text('analyse antivirus en cours', count: 1) end scenario 'retry on transcient upload error', js: true do log_in(user, procedure_with_pjs) fill_individual # Test auto-upload failure # Make the subsequent auto-upload request fail allow_any_instance_of(Champs::PieceJustificativeController).to receive(:update) do |instance| instance.render json: { errors: ['Error'] }, status: :bad_request end attach_file('Pièce justificative 1', Rails.root + 'spec/fixtures/files/file.pdf') expect(page).to have_text('Une erreur s’est produite pendant l’envoi du fichier') expect(page).to have_button('Ré-essayer', visible: true) expect(page).to have_button('Déposer le dossier', disabled: false) allow_any_instance_of(Champs::PieceJustificativeController).to receive(:update).and_call_original # Test that retrying after a failure works click_on('Ré-essayer', visible: true) expect(page).to have_text('analyse antivirus en cours') expect(page).to have_text('file.pdf') expect(page).to have_button('Déposer le dossier', disabled: false) # Reload the current page visit current_path # Expect the file to have been saved on the dossier expect(page).to have_text('file.pdf') end context 'with condition' do include Logic context 'with a required conditionnal champ' do let(:procedure) do procedure = create(:procedure, :published, :for_individual, types_de_champ_public: [ { type: :integer_number, libelle: 'age' }, { type: :text, libelle: 'nom', mandatory: true } ]) age, nom = procedure.draft_revision.types_de_champ.all nom.update(condition: greater_than_eq(champ_value(age.stable_id), constant(18))) procedure end scenario 'submit a dossier with an hidden mandatory champ ', js: true do log_in(user, procedure) fill_individual click_on 'Déposer le dossier' expect(page).to have_current_path(merci_dossier_path(user_dossier)) end scenario 'cannot submit a reveal dossier with a revealed mandatory champ ', js: true do log_in(user, procedure) fill_individual fill_in('age', with: '18') expect(page).to have_css('label', text: 'nom *', visible: :visible) click_on 'Déposer le dossier' expect(page).to have_current_path(brouillon_dossier_path(user_dossier)) end end context 'with a visibilite in cascade' do let(:procedure) do procedure = create(:procedure, :for_individual).tap do |p| p.draft_revision.add_type_de_champ(type_champ: :integer_number, libelle: 'age') p.draft_revision.add_type_de_champ(type_champ: :yes_no, libelle: 'permis de conduire') p.draft_revision.add_type_de_champ(type_champ: :integer_number, libelle: 'tonnage') p.draft_revision.add_type_de_champ(type_champ: :text, libelle: 'parking') end age, permis, tonnage, parking = procedure.draft_revision.types_de_champ.all permis.update(condition: greater_than_eq(champ_value(age.stable_id), constant(18))) tonnage.update(condition: ds_eq(champ_value(permis.stable_id), constant(true))) parking.update(condition: less_than_eq(champ_value(tonnage.stable_id), constant(20))) procedure.publish! procedure end scenario 'fill a dossier', js: true do log_in(user, procedure) fill_individual expect(page).to have_css('label', text: 'age', visible: true) expect(page).to have_no_css('label', text: 'permis de conduire', visible: true) expect(page).to have_no_css('label', text: 'tonnage', visible: true) fill_in('age', with: '18') expect(page).to have_css('label', text: 'permis de conduire', visible: true) expect(page).to have_no_css('label', text: 'tonnage', visible: true) choose('Oui') expect(page).to have_css('label', text: 'permis de conduire', visible: true) expect(page).to have_css('label', text: 'tonnage', visible: true) fill_in('tonnage', with: '1') expect(page).to have_css('label', text: 'parking', visible: true) # try to fill with invalid data fill_in('tonnage', with: 'a') expect(page).to have_no_css('label', text: 'parking', visible: true) fill_in('age', with: '2') expect(page).to have_no_css('label', text: 'permis de conduire', visible: true) expect(page).to have_no_css('label', text: 'tonnage', visible: true) click_on 'Déposer le dossier' click_on 'Accéder à votre dossier' click_on 'Modifier mon dossier' expect(page).to have_css('label', text: 'age', visible: true) expect(page).to have_no_css('label', text: 'permis de conduire', visible: true) expect(page).to have_no_css('label', text: 'tonnage', visible: true) fill_in('age', with: '18') # the champ keeps their previous value so they are all displayed expect(page).to have_css('label', text: 'permis de conduire', visible: true) expect(page).to have_css('label', text: 'tonnage', visible: true) end end end context 'draft autosave' do scenario 'autosave a draft', js: true do log_in(user, simple_procedure) fill_individual expect(page).to have_no_button('Enregistrer le brouillon') expect(page).to have_content('Votre brouillon est automatiquement enregistré') fill_in('texte obligatoire', with: 'a valid user input') blur expect(page).to have_css('span', text: 'Brouillon enregistré', visible: true) visit current_path expect(page).to have_field('texte obligatoire', with: 'a valid user input') end scenario 'retry on autosave error', :capybara_ignore_server_errors, js: true do log_in(user, simple_procedure) fill_individual # Test autosave failure allow_any_instance_of(Users::DossiersController).to receive(:update_brouillon).and_raise("Server is busy") fill_in('texte obligatoire', with: 'a valid user input') blur expect(page).to have_css('span', text: 'Impossible d’enregistrer le brouillon', visible: true) # Test that retrying after a failure works allow_any_instance_of(Users::DossiersController).to receive(:update_brouillon).and_call_original click_on 'réessayer' expect(page).to have_css('span', text: 'Brouillon enregistré', visible: true) visit current_path expect(page).to have_field('texte obligatoire', with: 'a valid user input') end scenario 'autosave redirects to sign-in after being disconnected', js: true do log_in(user, simple_procedure) fill_individual # When the user is disconnected # (either because signing-out in another tab, or because the session cookie expired) logout(:user) fill_in('texte obligatoire', with: 'a valid user input') blur # … they are redirected to the sign-in page. expect(page).to have_current_path(new_user_session_path) # After sign-in, they are redirected back to their brouillon sign_in_with(user.email, password) expect(page).to have_current_path(brouillon_dossier_path(user_dossier)) fill_in('texte obligatoire', with: 'a valid user input') blur expect(page).to have_css('span', text: 'Brouillon enregistré', visible: true) end end private def log_in(user, procedure) login_as user, scope: :user visit "/commencer/#{procedure.path}" click_on 'Commencer la démarche' expect(page).to have_content("Données d’identité") expect(page).to have_current_path(identite_dossier_path(user_dossier)) end def form_id_for_datetime(libelle) # The HTML for datetime is a bit specific since it has 5 selects, below here is a sample HTML # So, we want to find the partial id of a datetime (partial because there are 5 ids: # dossier_champs_attributes_3_value_1i, 2i, ... 5i) ; we are interested in the 'dossier_champs_attributes_3_value' part # which is then completed in select_date_and_time and check_date_and_time # # We find the H2, find the first select in the next .datetime div, then strip the last 3 characters # #