mirror of
https://github.com/Freika/dawarich.git
synced 2026-01-09 08:47:11 -05:00
* fix: move foreman to global gems to fix startup crash (#1971) * Update exporting code to stream points data to file in batches to red… (#1980) * Update exporting code to stream points data to file in batches to reduce memory usage * Update changelog * Update changelog * Feature/maplibre frontend (#1953) * Add a plan to use MapLibre GL JS for the frontend map rendering, replacing Leaflet * Implement phase 1 * Phases 1-3 + part of 4 * Fix e2e tests * Phase 6 * Implement fog of war * Phase 7 * Next step: fix specs, phase 7 done * Use our own map tiles * Extract v2 map logic to separate manager classes * Update settings panel on v2 map * Update v2 e2e tests structure * Reimplement location search in maps v2 * Update speed routes * Implement visits and places creation in v2 * Fix last failing test * Implement visits merging * Fix a routes e2e test and simplify the routes layer styling. * Extract js to modules from maps_v2_controller.js * Implement area creation * Fix spec problem * Fix some e2e tests * Implement live mode in v2 map * Update icons and panel * Extract some styles * Remove unused file * Start adding dark theme to popups on MapLibre maps * Make popups respect dark theme * Move v2 maps to maplibre namespace * Update v2 references to maplibre * Put place, area and visit info into side panel * Update API to use safe settings config method * Fix specs * Fix method name to config in SafeSettings and update usages accordingly * Add missing public files * Add handling for real time points * Fix remembering enabled/disabled layers of the v2 map * Fix lots of e2e tests * Add settings to select map version * Use maps/v2 as main path for MapLibre maps * Update routing * Update live mode * Update maplibre controller * Update changelog * Remove some console.log statements * Pull only necessary data for map v2 points * Feature/raw data archive (#2009) * 0.36.2 (#2007) * fix: move foreman to global gems to fix startup crash (#1971) * Update exporting code to stream points data to file in batches to red… (#1980) * Update exporting code to stream points data to file in batches to reduce memory usage * Update changelog * Update changelog * Feature/maplibre frontend (#1953) * Add a plan to use MapLibre GL JS for the frontend map rendering, replacing Leaflet * Implement phase 1 * Phases 1-3 + part of 4 * Fix e2e tests * Phase 6 * Implement fog of war * Phase 7 * Next step: fix specs, phase 7 done * Use our own map tiles * Extract v2 map logic to separate manager classes * Update settings panel on v2 map * Update v2 e2e tests structure * Reimplement location search in maps v2 * Update speed routes * Implement visits and places creation in v2 * Fix last failing test * Implement visits merging * Fix a routes e2e test and simplify the routes layer styling. * Extract js to modules from maps_v2_controller.js * Implement area creation * Fix spec problem * Fix some e2e tests * Implement live mode in v2 map * Update icons and panel * Extract some styles * Remove unused file * Start adding dark theme to popups on MapLibre maps * Make popups respect dark theme * Move v2 maps to maplibre namespace * Update v2 references to maplibre * Put place, area and visit info into side panel * Update API to use safe settings config method * Fix specs * Fix method name to config in SafeSettings and update usages accordingly * Add missing public files * Add handling for real time points * Fix remembering enabled/disabled layers of the v2 map * Fix lots of e2e tests * Add settings to select map version * Use maps/v2 as main path for MapLibre maps * Update routing * Update live mode * Update maplibre controller * Update changelog * Remove some console.log statements --------- Co-authored-by: Robin Tuszik <mail@robin.gg> * Remove esbuild scripts from package.json * Remove sideEffects field from package.json * Raw data archivation * Add tests * Fix tests * Fix tests * Update ExceptionReporter * Add schedule to run raw data archival job monthly * Change file structure for raw data archival feature * Update changelog and version for raw data archival feature --------- Co-authored-by: Robin Tuszik <mail@robin.gg> * Set raw_data to an empty hash instead of nil when archiving * Fix storage configuration and file extraction * Consider MIN_MINUTES_SPENT_IN_CITY during stats calculation (#2018) * Consider MIN_MINUTES_SPENT_IN_CITY during stats calculation * Remove raw data from visited cities api endpoint * Use user timezone to show dates on maps (#2020) * Fix/pre epoch time (#2019) * Use user timezone to show dates on maps * Limit timestamps to valid range to prevent database errors when users enter pre-epoch dates. * Limit timestamps to valid range to prevent database errors when users enter pre-epoch dates. * Fix tests failing due to new index on stats table * Fix failing specs * Update redis client configuration to support unix socket connection * Update changelog * Fix kml kmz import issues (#2023) * Fix kml kmz import issues * Refactor KML importer to improve readability and maintainability * Implement moving points in map v2 and fix route rendering logic to ma… (#2027) * Implement moving points in map v2 and fix route rendering logic to match map v1. * Fix route spec * fix(maplibre): update date format to ISO 8601 (#2029) * Add verification step to raw data archival process (#2028) * Add verification step to raw data archival process * Add actual verification of raw data archives after creation, and only clear raw_data for verified archives. * Fix failing specs * Eliminate zip-bomb risk * Fix potential memory leak in js * Return .keep files * Use Toast instead of alert for notifications * Add help section to navbar dropdown * Update changelog * Remove raw_data_archival_job * Ensure file is being closed properly after reading in Archivable concern --------- Co-authored-by: Robin Tuszik <mail@robin.gg>
206 lines
7.7 KiB
Ruby
206 lines
7.7 KiB
Ruby
# frozen_string_literal: true
|
|
|
|
require 'rails_helper'
|
|
|
|
RSpec.describe Stats::CalculateMonth do
|
|
describe '#call' do
|
|
subject(:calculate_stats) { described_class.new(user.id, year, month).call }
|
|
|
|
let(:user) { create(:user) }
|
|
let(:year) { 2021 }
|
|
let(:month) { 1 }
|
|
|
|
context 'when there are no points' do
|
|
it 'does not create stats' do
|
|
expect { calculate_stats }.not_to(change { Stat.count })
|
|
end
|
|
|
|
context 'when stats already exist for the month' do
|
|
before do
|
|
create(:stat, user: user, year: year, month: month)
|
|
end
|
|
|
|
it 'deletes existing stats for that month' do
|
|
expect { calculate_stats }.to change { Stat.count }.by(-1)
|
|
end
|
|
end
|
|
end
|
|
|
|
context 'when there are points' do
|
|
let(:timestamp1) { DateTime.new(year, month, 1, 12).to_i }
|
|
let(:timestamp2) { DateTime.new(year, month, 1, 13).to_i }
|
|
let(:timestamp3) { DateTime.new(year, month, 1, 14).to_i }
|
|
let!(:import) { create(:import, user:) }
|
|
let!(:point1) do
|
|
create(:point,
|
|
user:,
|
|
import:,
|
|
timestamp: timestamp1,
|
|
lonlat: 'POINT(14.452712811406352 52.107902115161316)')
|
|
end
|
|
let!(:point2) do
|
|
create(:point,
|
|
user:,
|
|
import:,
|
|
timestamp: timestamp2,
|
|
lonlat: 'POINT(12.291519487061901 51.9746598171507)')
|
|
end
|
|
let!(:point3) do
|
|
create(:point,
|
|
user:,
|
|
import:,
|
|
timestamp: timestamp3,
|
|
lonlat: 'POINT(9.77973105800526 52.72859111523629)')
|
|
end
|
|
|
|
context 'when calculating distance' do
|
|
it 'creates stats' do
|
|
expect { calculate_stats }.to change { Stat.count }.by(1)
|
|
end
|
|
|
|
it 'calculates distance in meters consistently' do
|
|
calculate_stats
|
|
|
|
# Distance should be calculated in meters regardless of user unit preference
|
|
# The actual distance between the test points is approximately 340 km = 340,000 meters
|
|
expect(user.stats.last.distance).to be_within(1000).of(340_000)
|
|
end
|
|
|
|
context 'when there is an error' do
|
|
before do
|
|
allow(Stat).to receive(:find_or_initialize_by).and_raise(StandardError)
|
|
end
|
|
|
|
it 'does not create stats' do
|
|
expect { calculate_stats }.not_to(change { Stat.count })
|
|
end
|
|
|
|
it 'creates a notification' do
|
|
expect { calculate_stats }.to change { Notification.count }.by(1)
|
|
end
|
|
end
|
|
end
|
|
|
|
context 'when user prefers miles' do
|
|
before do
|
|
user.update(settings: { maps: { distance_unit: 'mi' } })
|
|
end
|
|
|
|
it 'still stores distance in meters (same as km users)' do
|
|
calculate_stats
|
|
|
|
# Distance stored should be the same regardless of user preference (meters)
|
|
expect(user.stats.last.distance).to be_within(1000).of(340_000)
|
|
end
|
|
end
|
|
|
|
context 'when calculating visited cities and countries' do
|
|
let(:timestamp_base) { DateTime.new(year, month, 1, 12).to_i }
|
|
let!(:import) { create(:import, user:) }
|
|
|
|
context 'when user spent more than MIN_MINUTES_SPENT_IN_CITY in a city' do
|
|
let!(:berlin_points) do
|
|
[
|
|
create(:point, user:, import:, timestamp: timestamp_base,
|
|
city: 'Berlin', country_name: 'Germany',
|
|
lonlat: 'POINT(13.404954 52.520008)'),
|
|
create(:point, user:, import:, timestamp: timestamp_base + 30.minutes,
|
|
city: 'Berlin', country_name: 'Germany',
|
|
lonlat: 'POINT(13.404954 52.520008)'),
|
|
create(:point, user:, import:, timestamp: timestamp_base + 70.minutes,
|
|
city: 'Berlin', country_name: 'Germany',
|
|
lonlat: 'POINT(13.404954 52.520008)')
|
|
]
|
|
end
|
|
|
|
it 'includes the city in toponyms' do
|
|
calculate_stats
|
|
|
|
stat = user.stats.last
|
|
expect(stat.toponyms).not_to be_empty
|
|
expect(stat.toponyms.first['country']).to eq('Germany')
|
|
expect(stat.toponyms.first['cities']).not_to be_empty
|
|
expect(stat.toponyms.first['cities'].first['city']).to eq('Berlin')
|
|
end
|
|
end
|
|
|
|
context 'when user spent less than MIN_MINUTES_SPENT_IN_CITY in a city' do
|
|
let!(:prague_points) do
|
|
[
|
|
create(:point, user:, import:, timestamp: timestamp_base,
|
|
city: 'Prague', country_name: 'Czech Republic',
|
|
lonlat: 'POINT(14.4378 50.0755)'),
|
|
create(:point, user:, import:, timestamp: timestamp_base + 10.minutes,
|
|
city: 'Prague', country_name: 'Czech Republic',
|
|
lonlat: 'POINT(14.4378 50.0755)'),
|
|
create(:point, user:, import:, timestamp: timestamp_base + 20.minutes,
|
|
city: 'Prague', country_name: 'Czech Republic',
|
|
lonlat: 'POINT(14.4378 50.0755)')
|
|
]
|
|
end
|
|
|
|
it 'excludes the city from toponyms' do
|
|
calculate_stats
|
|
|
|
stat = user.stats.last
|
|
expect(stat.toponyms).not_to be_empty
|
|
|
|
# Country should be listed but with no cities
|
|
czech_country = stat.toponyms.find { |t| t['country'] == 'Czech Republic' }
|
|
expect(czech_country).not_to be_nil
|
|
expect(czech_country['cities']).to be_empty
|
|
end
|
|
end
|
|
|
|
context 'when user visited multiple cities with mixed durations' do
|
|
let!(:mixed_points) do
|
|
[
|
|
# Berlin: 70 minutes (should be included)
|
|
create(:point, user:, import:, timestamp: timestamp_base,
|
|
city: 'Berlin', country_name: 'Germany',
|
|
lonlat: 'POINT(13.404954 52.520008)'),
|
|
create(:point, user:, import:, timestamp: timestamp_base + 70.minutes,
|
|
city: 'Berlin', country_name: 'Germany',
|
|
lonlat: 'POINT(13.404954 52.520008)'),
|
|
|
|
# Prague: 20 minutes (should be excluded)
|
|
create(:point, user:, import:, timestamp: timestamp_base + 100.minutes,
|
|
city: 'Prague', country_name: 'Czech Republic',
|
|
lonlat: 'POINT(14.4378 50.0755)'),
|
|
create(:point, user:, import:, timestamp: timestamp_base + 120.minutes,
|
|
city: 'Prague', country_name: 'Czech Republic',
|
|
lonlat: 'POINT(14.4378 50.0755)'),
|
|
|
|
# Vienna: 90 minutes (should be included)
|
|
create(:point, user:, import:, timestamp: timestamp_base + 150.minutes,
|
|
city: 'Vienna', country_name: 'Austria',
|
|
lonlat: 'POINT(16.3738 48.2082)'),
|
|
create(:point, user:, import:, timestamp: timestamp_base + 240.minutes,
|
|
city: 'Vienna', country_name: 'Austria',
|
|
lonlat: 'POINT(16.3738 48.2082)')
|
|
]
|
|
end
|
|
|
|
it 'only includes cities where user spent >= MIN_MINUTES_SPENT_IN_CITY' do
|
|
calculate_stats
|
|
|
|
stat = user.stats.last
|
|
expect(stat.toponyms).not_to be_empty
|
|
|
|
# Get all cities from all countries
|
|
all_cities = stat.toponyms.flat_map { |t| t['cities'].map { |c| c['city'] } }
|
|
|
|
# Berlin and Vienna should be included
|
|
expect(all_cities).to include('Berlin', 'Vienna')
|
|
|
|
# Prague should NOT be included
|
|
expect(all_cities).not_to include('Prague')
|
|
|
|
# Should have exactly 2 cities
|
|
expect(all_cities.size).to eq(2)
|
|
end
|
|
end
|
|
end
|
|
end
|
|
end
|
|
end
|