dawarich/spec/services/users/import_data/imports_spec.rb

282 lines
8.6 KiB
Ruby
Raw Normal View History

2025-06-28 06:22:56 -04:00
# frozen_string_literal: true
require 'rails_helper'
RSpec.describe Users::ImportData::Imports, type: :service do
let(:user) { create(:user) }
let(:files_directory) { Rails.root.join('tmp', 'test_files') }
let(:imports_data) do
[
{
'name' => '2023_MARCH.json',
'source' => 'google_semantic_history',
'created_at' => '2024-01-01T00:00:00Z',
'updated_at' => '2024-01-01T00:00:00Z',
'processed' => true,
'file_name' => 'import_1_2023_MARCH.json',
'original_filename' => '2023_MARCH.json',
'file_size' => 2048576,
'content_type' => 'application/json'
},
{
'name' => '2023_APRIL.json',
'source' => 'owntracks',
'created_at' => '2024-01-02T00:00:00Z',
'updated_at' => '2024-01-02T00:00:00Z',
'processed' => false,
'file_name' => 'import_2_2023_APRIL.json',
'original_filename' => '2023_APRIL.json',
'file_size' => 1048576,
'content_type' => 'application/json'
}
]
end
let(:service) { described_class.new(user, imports_data, files_directory) }
before do
FileUtils.mkdir_p(files_directory)
# Create mock files
File.write(files_directory.join('import_1_2023_MARCH.json'), '{"test": "data"}')
File.write(files_directory.join('import_2_2023_APRIL.json'), '{"more": "data"}')
end
after do
FileUtils.rm_rf(files_directory) if files_directory.exist?
end
describe '#call' do
context 'with valid imports data' do
it 'creates new imports for the user' do
expect { service.call }.to change { user.imports.count }.by(2)
end
it 'creates imports with correct attributes' do
service.call
march_import = user.imports.find_by(name: '2023_MARCH.json')
expect(march_import).to have_attributes(
name: '2023_MARCH.json',
source: 'google_semantic_history',
processed: 1
)
april_import = user.imports.find_by(name: '2023_APRIL.json')
expect(april_import).to have_attributes(
name: '2023_APRIL.json',
source: 'owntracks',
processed: 0
)
end
it 'attaches files to the imports' do
service.call
march_import = user.imports.find_by(name: '2023_MARCH.json')
expect(march_import.file).to be_attached
expect(march_import.file.filename.to_s).to eq('2023_MARCH.json')
expect(march_import.file.content_type).to eq('application/json')
april_import = user.imports.find_by(name: '2023_APRIL.json')
expect(april_import.file).to be_attached
expect(april_import.file.filename.to_s).to eq('2023_APRIL.json')
expect(april_import.file.content_type).to eq('application/json')
end
it 'returns the number of imports and files created' do
imports_created, files_restored = service.call
expect(imports_created).to eq(2)
expect(files_restored).to eq(2)
end
it 'logs the import process' do
allow(Rails.logger).to receive(:info) # Allow all info logs (including ActiveStorage)
expect(Rails.logger).to receive(:info).with("Importing 2 imports for user: #{user.email}")
expect(Rails.logger).to receive(:info).with("Imports import completed. Created: 2, Files restored: 2")
service.call
end
2025-06-30 14:29:47 -04:00
it 'does not trigger background processing jobs' do
expect(Import::ProcessJob).not_to receive(:perform_later)
service.call
end
it 'sets skip_background_processing flag on created imports' do
service.call
user.imports.each do |import|
expect(import.skip_background_processing).to be_truthy
end
end
2025-06-28 06:22:56 -04:00
end
context 'with duplicate imports' do
before do
# Create an existing import with same name, source, and created_at
user.imports.create!(
name: '2023_MARCH.json',
source: 'google_semantic_history',
created_at: Time.parse('2024-01-01T00:00:00Z')
)
end
it 'skips duplicate imports' do
expect { service.call }.to change { user.imports.count }.by(1)
end
it 'logs when skipping duplicates' do
allow(Rails.logger).to receive(:debug) # Allow any debug logs
expect(Rails.logger).to receive(:debug).with("Import already exists: 2023_MARCH.json")
service.call
end
it 'returns only the count of newly created imports' do
imports_created, files_restored = service.call
expect(imports_created).to eq(1)
expect(files_restored).to eq(1)
end
end
context 'with missing files' do
before do
FileUtils.rm_f(files_directory.join('import_1_2023_MARCH.json'))
end
it 'creates imports but logs file errors' do
expect(Rails.logger).to receive(:warn).with(/Import file not found/)
imports_created, files_restored = service.call
expect(imports_created).to eq(2)
expect(files_restored).to eq(1) # Only one file was successfully restored
end
it 'creates imports without file attachments for missing files' do
service.call
march_import = user.imports.find_by(name: '2023_MARCH.json')
expect(march_import.file).not_to be_attached
end
end
context 'with imports that have no files (null file_name)' do
let(:imports_data) do
[
{
'name' => 'No File Import',
'source' => 'gpx',
'created_at' => '2024-01-01T00:00:00Z',
'processed' => true,
'file_name' => nil,
'original_filename' => nil
}
]
end
it 'creates imports without attempting file restoration' do
expect { service.call }.to change { user.imports.count }.by(1)
end
it 'returns correct counts' do
imports_created, files_restored = service.call
expect(imports_created).to eq(1)
expect(files_restored).to eq(0)
end
end
context 'with invalid import data' do
let(:imports_data) do
[
{ 'name' => 'Valid Import', 'source' => 'owntracks' },
'invalid_data',
{ 'name' => 'Another Valid Import', 'source' => 'gpx' }
]
end
it 'skips invalid entries and imports valid ones' do
expect { service.call }.to change { user.imports.count }.by(2)
end
it 'returns the count of valid imports created' do
imports_created, files_restored = service.call
expect(imports_created).to eq(2)
expect(files_restored).to eq(0) # No files for these imports
end
end
context 'with validation errors' do
let(:imports_data) do
[
{ 'name' => 'Valid Import', 'source' => 'owntracks' },
{ 'source' => 'owntracks' }, # missing name
{ 'name' => 'Missing Source Import' } # missing source
]
end
it 'only creates valid imports' do
expect { service.call }.to change { user.imports.count }.by(2)
# Verify only the valid imports were created (name is required, source defaults to first enum)
created_imports = user.imports.pluck(:name, :source)
expect(created_imports).to contain_exactly(
['Valid Import', 'owntracks'],
['Missing Source Import', 'google_semantic_history']
)
end
it 'logs validation errors' do
expect(Rails.logger).to receive(:error).at_least(:once)
service.call
end
end
context 'with nil imports data' do
let(:imports_data) { nil }
it 'does not create any imports' do
expect { service.call }.not_to change { user.imports.count }
end
it 'returns [0, 0]' do
result = service.call
expect(result).to eq([0, 0])
end
end
context 'with non-array imports data' do
let(:imports_data) { 'invalid_data' }
it 'does not create any imports' do
expect { service.call }.not_to change { user.imports.count }
end
it 'returns [0, 0]' do
result = service.call
expect(result).to eq([0, 0])
end
end
context 'with empty imports data' do
let(:imports_data) { [] }
it 'does not create any imports' do
expect { service.call }.not_to change { user.imports.count }
end
it 'logs the import process with 0 count' do
expect(Rails.logger).to receive(:info).with("Importing 0 imports for user: #{user.email}")
expect(Rails.logger).to receive(:info).with("Imports import completed. Created: 0, Files restored: 0")
service.call
end
it 'returns [0, 0]' do
result = service.call
expect(result).to eq([0, 0])
end
end
end
end