123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680 |
- require 'rails_helper'
- require 'models/application_model_examples'
- RSpec.describe Job, type: :model do
- it_behaves_like 'ApplicationModel', can_assets: { selectors: %i[condition perform] }
- subject(:job) { create(:job) }
- describe 'Class methods:' do
- describe '.run' do
- let!(:executable_jobs) { jobs.select(&:executable?).select(&:in_timeplan?) }
- let!(:nonexecutable_jobs) { jobs - executable_jobs }
- let!(:jobs) do
- [
- # executable
- create(:job, :always_on, updated_at: 2.minutes.ago),
- # not executable (updated too recently)
- create(:job),
- # not executable (inactive)
- create(:job, updated_at: 2.minutes.ago, active: false),
- # not executable (started too recently)
- create(:job, :always_on, updated_at: 2.minutes.ago, last_run_at: 5.minutes.ago),
- # executable
- create(:job, :always_on, updated_at: 2.minutes.ago, last_run_at: 15.minutes.ago),
- # not executable (still running, started too recently)
- create(:job, :always_on, updated_at: 2.minutes.ago, running: true, last_run_at: 23.hours.ago),
- # executable
- create(:job, :always_on, updated_at: 2.minutes.ago, running: true, last_run_at: 25.hours.ago),
- ]
- end
- it 'runs all executable jobs (and no others)' do
- expect { described_class.run }
- .to change { executable_jobs.map(&:reload).map(&:last_run_at).any?(&:nil?) }.to(false)
- .and not_change { nonexecutable_jobs.map(&:reload).map(&:last_run_at).all?(&:nil?) }
- end
- end
- end
- describe 'Instance methods:' do
- describe '#run' do
- subject(:job) { create(:job, condition: condition, perform: perform) }
- let(:condition) do
- {
- 'ticket.state_id' => {
- 'operator' => 'is',
- 'value' => Ticket::State.where(name: %w[new open]).pluck(:id).map(&:to_s)
- },
- 'ticket.created_at' => {
- 'operator' => 'before (relative)',
- 'value' => '2',
- 'range' => 'day'
- },
- }
- end
- let(:perform) do
- { 'ticket.state_id' => { 'value' => Ticket::State.find_by(name: 'closed').id.to_s } }
- end
- let!(:matching_ticket) do
- create(:ticket, state: Ticket::State.lookup(name: 'new'), created_at: 3.days.ago)
- end
- let!(:nonmatching_ticket) do
- create(:ticket, state: Ticket::State.lookup(name: 'new'), created_at: 1.day.ago)
- end
- context 'when job is not #executable?' do
- before { allow(job).to receive(:executable?).and_return(false) }
- it 'does not perform changes on matching tickets' do
- expect { job.run }.not_to change { matching_ticket.reload.state }
- end
- it 'does not update #last_run_at' do
- expect { job.run }.to not_change { job.reload.last_run_at }
- end
- context 'but "force" flag is given' do
- it 'performs changes on matching tickets' do
- expect { job.run(true) }
- .to change { matching_ticket.reload.state }
- .and not_change { nonmatching_ticket.reload.state }
- end
- it 'updates #last_run_at' do
- expect { job.run(true) }
- .to change { job.reload.last_run_at }
- end
- end
- end
- context 'when job is #executable?' do
- before { allow(job).to receive(:executable?).and_return(true) }
- context 'and due (#in_timeplan?)' do
- before { allow(job).to receive(:in_timeplan?).and_return(true) }
- it 'updates #last_run_at' do
- expect { job.run }.to change { job.reload.last_run_at }
- end
- it 'performs changes on matching tickets' do
- expect { job.run }
- .to change { matching_ticket.reload.state }
- .and not_change { nonmatching_ticket.reload.state }
- end
- context 'and already marked #running' do
- before { job.update(running: true) }
- it 'resets #running to false' do
- expect { job.run }.to change { job.reload.running }.to(false)
- end
- end
- context 'but condition doesn’t match any tickets' do
- before { job.update(condition: invalid_condition) }
- let(:invalid_condition) do
- { 'ticket.state_id' => { 'operator' => 'is', 'value' => '9999' } }
- end
- it 'performs no changes' do
- expect { job.run }
- .not_to change { matching_ticket.reload.state }
- end
- end
- describe 'use case: deleting tickets based on tag' do
- let(:condition) { { 'ticket.tags' => { 'operator' => 'contains one', 'value' => 'spam' } } }
- let(:perform) { { 'ticket.action' => { 'value' => 'delete' } } }
- let!(:matching_ticket) { create(:ticket).tap { |t| t.tag_add('spam', 1) } }
- let!(:nonmatching_ticket) { create(:ticket) }
- it 'deletes tickets matching the specified tags' do
- job.run
- expect { matching_ticket.reload }.to raise_error(ActiveRecord::RecordNotFound)
- expect { nonmatching_ticket.reload }.not_to raise_error
- end
- end
- describe 'use case: deleting tickets based on group' do
- let(:condition) { { 'ticket.group_id' => { 'operator' => 'is', 'value' => matching_ticket.group.id } } }
- let(:perform) { { 'ticket.action' => { 'value' => 'delete' } } }
- let!(:matching_ticket) { create(:ticket) }
- let!(:nonmatching_ticket) { create(:ticket) }
- it 'deletes tickets matching the specified groups' do
- job.run
- expect { matching_ticket.reload }.to raise_error(ActiveRecord::RecordNotFound)
- expect { nonmatching_ticket.reload }.not_to raise_error
- end
- end
- end
- context 'and not due yet' do
- before { allow(job).to receive(:in_timeplan?).and_return(false) }
- it 'does not perform changes on matching tickets' do
- expect { job.run }.not_to change { matching_ticket.reload.state }
- end
- it 'does not update #last_run_at' do
- expect { job.run }.to not_change { job.reload.last_run_at }
- end
- it 'updates #next_run_at' do
- travel_to(Time.current.last_week) # force new value for #next_run_at
- expect { job.run }.to change { job.reload.next_run_at }
- end
- context 'but "force" flag is given' do
- it 'performs changes on matching tickets' do
- expect { job.run(true) }
- .to change { matching_ticket.reload.state }
- .and not_change { nonmatching_ticket.reload.state }
- end
- it 'updates #last_run_at' do
- expect { job.run(true) }.to change { job.reload.last_run_at }
- end
- it 'updates #next_run_at' do
- travel_to(Time.current.last_week) # force new value for #next_run_at
- expect { job.run }.to change { job.reload.next_run_at }
- end
- end
- end
- end
- context 'when job has pre_condition:current_user.id in selector' do
- let!(:matching_ticket) { create(:ticket, owner_id: 1) }
- let!(:nonmatching_ticket) { create(:ticket, owner_id: create(:agent_user).id) }
- let(:condition) do
- {
- 'ticket.owner_id' => {
- 'operator' => 'is',
- 'pre_condition' => 'current_user.id',
- 'value' => '',
- 'value_completion' => ''
- },
- }
- end
- before do
- UserInfo.current_user_id = create(:admin_user).id
- job
- UserInfo.current_user_id = nil
- end
- it 'performs changes on matching tickets' do
- expect { job.run(true) }
- .to change { matching_ticket.reload.state }
- .and not_change { nonmatching_ticket.reload.state }
- end
- end
- end
- describe '#executable?' do
- context 'for an inactive Job' do
- subject(:job) { create(:job, active: false) }
- it 'returns false' do
- expect(job.executable?).to be(false)
- end
- end
- context 'for an active job' do
- context 'updated in the last minute' do
- subject(:job) do
- create(:job, active: true,
- updated_at: 59.seconds.ago)
- end
- it 'returns false' do
- expect(job.executable?).to be(false)
- end
- end
- context 'updated over a minute ago' do
- context 'that has never run before' do
- subject(:job) do
- create(:job, active: true,
- updated_at: 60.seconds.ago)
- end
- it 'returns true' do
- expect(job.executable?).to be(true)
- end
- end
- context 'that was started in the last 10 minutes' do
- subject(:job) do
- create(:job, active: true,
- updated_at: 60.seconds.ago,
- last_run_at: 9.minutes.ago)
- end
- it 'returns false' do
- expect(job.executable?).to be(false)
- end
- context '(or, given an argument, up to 10 minutes before that)' do
- subject(:job) do
- create(:job, active: true,
- updated_at: 60.seconds.ago,
- last_run_at: 9.minutes.before(Time.current.yesterday))
- end
- it 'returns false' do
- expect(job.executable?(Time.current.yesterday)).to be(false)
- end
- end
- end
- context 'that was started over 10 minutes ago' do
- subject(:job) do
- create(:job, active: true,
- updated_at: 60.seconds.ago,
- last_run_at: 10.minutes.ago)
- end
- it 'returns true' do
- expect(job.executable?).to be(true)
- end
- context '(or, given an argument, over 10 minutes before that)' do
- subject(:job) do
- create(:job, active: true,
- updated_at: 60.seconds.ago,
- last_run_at: 10.minutes.before(Time.current.yesterday))
- end
- it 'returns true' do
- expect(job.executable?(Time.current.yesterday)).to be(true)
- end
- end
- context 'but is still running, up to 24 hours later' do
- subject(:job) do
- create(:job, active: true,
- updated_at: 60.seconds.ago,
- running: true,
- last_run_at: 23.hours.ago)
- end
- it 'returns false' do
- expect(job.executable?).to be(false)
- end
- end
- context 'but is still running, over 24 hours later' do
- subject(:job) do
- create(:job, active: true,
- updated_at: 60.seconds.ago,
- running: true,
- last_run_at: 24.hours.ago)
- end
- it 'returns true' do
- expect(job.executable?).to be(true)
- end
- end
- end
- end
- end
- end
- describe '#in_timeplan?' do
- subject(:job) { create(:job, :never_on) }
- context 'when the current day, hour, and minute all match true values in #timeplan' do
- context 'for Symbol/Integer keys' do
- before do
- job.update(
- timeplan: {
- days: job.timeplan[:days]
- .transform_keys(&:to_sym)
- .merge(Time.current.strftime('%a').to_sym => true),
- hours: job.timeplan[:hours]
- .transform_keys(&:to_i)
- .merge(Time.current.hour => true),
- minutes: job.timeplan[:minutes]
- .transform_keys(&:to_i)
- .merge(Time.current.min.floor(-1) => true),
- }
- )
- end
- it 'returns true' do
- expect(job.in_timeplan?).to be(true)
- end
- end
- context 'for String keys' do
- before do
- job.update(
- timeplan: {
- days: job.timeplan[:days]
- .transform_keys(&:to_s)
- .merge(Time.current.strftime('%a') => true),
- hours: job.timeplan[:hours]
- .transform_keys(&:to_s)
- .merge(Time.current.hour.to_s => true),
- minutes: job.timeplan[:minutes]
- .transform_keys(&:to_s)
- .merge(Time.current.min.floor(-1).to_s => true),
- }
- )
- end
- it 'returns true' do
- expect(job.in_timeplan?).to be(true)
- end
- end
- end
- context 'when the current day does not match a true value in #timeplan' do
- context 'for Symbol/Integer keys' do
- before do
- job.update(
- timeplan: {
- days: job.timeplan[:days]
- .transform_keys(&:to_sym)
- .transform_values { true }
- .merge(Time.current.strftime('%a').to_sym => false),
- hours: job.timeplan[:hours]
- .transform_keys(&:to_i)
- .merge(Time.current.hour => true),
- minutes: job.timeplan[:minutes]
- .transform_keys(&:to_i)
- .merge(Time.current.min.floor(-1) => true),
- }
- )
- end
- it 'returns false' do
- expect(job.in_timeplan?).to be(false)
- end
- end
- context 'for String keys' do
- before do
- job.update(
- timeplan: {
- days: job.timeplan[:days]
- .transform_keys(&:to_s)
- .transform_values { true }
- .merge(Time.current.strftime('%a') => false),
- hours: job.timeplan[:hours]
- .transform_keys(&:to_s)
- .merge(Time.current.hour.to_s => true),
- minutes: job.timeplan[:minutes]
- .transform_keys(&:to_s)
- .merge(Time.current.min.floor(-1).to_s => true),
- }
- )
- end
- it 'returns false' do
- expect(job.in_timeplan?).to be(false)
- end
- end
- end
- context 'when the current hour does not match a true value in #timeplan' do
- context 'for Symbol/Integer keys' do
- before do
- job.update(
- timeplan: {
- days: job.timeplan[:days]
- .transform_keys(&:to_sym)
- .merge(Time.current.strftime('%a').to_sym => true),
- hours: job.timeplan[:hours]
- .transform_keys(&:to_i)
- .transform_values { true }
- .merge(Time.current.hour => false),
- minutes: job.timeplan[:minutes]
- .transform_keys(&:to_i)
- .merge(Time.current.min.floor(-1) => true),
- }
- )
- end
- it 'returns false' do
- expect(job.in_timeplan?).to be(false)
- end
- end
- context 'for String keys' do
- before do
- job.update(
- timeplan: {
- days: job.timeplan[:days]
- .transform_keys(&:to_s)
- .merge(Time.current.strftime('%a') => true),
- hours: job.timeplan[:hours]
- .transform_keys(&:to_s)
- .transform_values { true }
- .merge(Time.current.hour.to_s => false),
- minutes: job.timeplan[:minutes]
- .transform_keys(&:to_s)
- .merge(Time.current.min.floor(-1).to_s => true),
- }
- )
- end
- it 'returns false' do
- expect(job.in_timeplan?).to be(false)
- end
- end
- end
- context 'when the current minute does not match a true value in #timeplan' do
- context 'for Symbol/Integer keys' do
- before do
- job.update(
- timeplan: {
- days: job.timeplan[:days]
- .transform_keys(&:to_sym)
- .merge(Time.current.strftime('%a').to_sym => true),
- hours: job.timeplan[:hours]
- .transform_keys(&:to_i)
- .merge(Time.current.hour => true),
- minutes: job.timeplan[:minutes]
- .transform_keys(&:to_i)
- .transform_values { true }
- .merge(Time.current.min.floor(-1) => false),
- }
- )
- end
- it 'returns false' do
- expect(job.in_timeplan?).to be(false)
- end
- end
- context 'for String keys' do
- before do
- job.update(
- timeplan: {
- days: job.timeplan[:days]
- .transform_keys(&:to_s)
- .merge(Time.current.strftime('%a') => true),
- hours: job.timeplan[:hours]
- .transform_keys(&:to_s)
- .merge(Time.current.hour.to_s => true),
- minutes: job.timeplan[:minutes]
- .transform_keys(&:to_s)
- .transform_values { true }
- .merge(Time.current.min.floor(-1).to_s => false),
- }
- )
- end
- it 'returns false' do
- expect(job.in_timeplan?).to be(false)
- end
- end
- end
- end
- end
- describe 'Attributes:' do
- describe '#next_run_at' do
- subject(:job) { build(:job) }
- it 'is set automatically on save (cannot be set manually)' do
- job.next_run_at = 1.day.from_now
- expect { job.save }.to change(job, :next_run_at)
- end
- context 'for an inactive Job' do
- subject(:job) { build(:job, active: false) }
- it 'is nil' do
- expect { job.save }
- .not_to change(job, :next_run_at).from(nil)
- end
- end
- context 'for a never-on Job (all #timeplan values are false)' do
- subject(:job) { build(:job, :never_on) }
- it 'is nil' do
- expect { job.save }
- .not_to change(job, :next_run_at).from(nil)
- end
- end
- context 'when #timeplan contains at least one true value for :day, :hour, and :minute' do
- subject(:job) { build(:job, :never_on) }
- let(:base_time) { Time.current.beginning_of_week }
- # Tuesday & Thursday @ 12:00a, 12:30a, 6:00p, and 6:30p
- before do
- job.assign_attributes(
- timeplan: {
- days: job.timeplan[:days].merge(Tue: true, Thu: true),
- hours: job.timeplan[:hours].merge(0 => true, 18 => true),
- minutes: job.timeplan[:minutes].merge(0 => true, 30 => true),
- }
- )
- end
- let(:valid_timeslots) do
- [
- base_time + 1.day, # Tue 12:00a
- base_time + 1.day + 30.minutes, # Tue 12:30a
- base_time + 1.day + 18.hours, # Tue 6:00p
- base_time + 1.day + 18.hours + 30.minutes, # Tue 6:30p
- base_time + 3.days, # Thu 12:00a
- base_time + 3.days + 30.minutes, # Thu 12:30a
- base_time + 3.days + 18.hours, # Thu 6:00p
- base_time + 3.days + 18.hours + 30.minutes, # Thu 6:30p
- ]
- end
- context 'for a Job that has never been run before' do
- context 'when record is saved at the start of the week' do
- before { travel_to(base_time) }
- it 'is set to the first valid timeslot of the week' do
- expect { job.save }
- .to change { job.next_run_at.to_i } # comparing times is hard;
- .to(valid_timeslots.first.to_i) # integers are less precise
- end
- end
- context 'when record is saved between two valid timeslots' do
- before { travel_to(valid_timeslots.third - 1.second) }
- it 'is set to the latter timeslot' do
- expect { job.save }
- .to change { job.next_run_at.to_i } # comparing times is hard;
- .to(valid_timeslots.third.to_i) # integers are less precise
- end
- end
- context 'when record is saved during a valid timeslot' do
- before { travel_to(valid_timeslots.fifth + 9.minutes + 59.seconds) }
- it 'is set to that timeslot' do
- expect { job.save }
- .to change { job.next_run_at.to_i } # comparing times is hard;
- .to(valid_timeslots.fifth.to_i) # integers are less precise
- end
- end
- end
- context 'for a Job that been run before' do
- context 'when record is saved in the same timeslot as #last_run_at' do
- before do
- job.assign_attributes(last_run_at: valid_timeslots.fourth + 5.minutes)
- travel_to(valid_timeslots.fourth + 7.minutes)
- end
- it 'is set to the next valid timeslot' do
- expect { job.save }
- .to change { job.next_run_at.to_i } # comparing times is hard;
- .to(valid_timeslots.fifth.to_i) # integers are less precise
- end
- end
- end
- end
- end
- describe '#perform' do
- describe 'Validations:' do
- describe '"article.note" key' do
- let(:perform) do
- { 'article.note' => { 'subject' => 'foo', 'internal' => 'true', 'body' => '' } }
- end
- it 'fails if an empty "body" is given' do
- expect { create(:job, perform: perform) }.to raise_error(Exceptions::UnprocessableEntity)
- end
- end
- describe '"notification.email" key' do
- let(:perform) do
- { 'notification.email' => { 'body' => 'foo', 'recipient' => '', 'subject' => 'bar' } }
- end
- it 'fails if an empty "recipient" is given' do
- expect { create(:job, perform: perform) }.to raise_error(Exceptions::UnprocessableEntity)
- end
- end
- describe '"notification.sms" key' do
- let(:perform) do
- { 'notification.sms' => { 'body' => 'foo', 'recipient' => '' } }
- end
- it 'fails if an empty "recipient" is given' do
- expect { create(:job, perform: perform) }.to raise_error(Exceptions::UnprocessableEntity)
- end
- end
- end
- end
- end
- end
|