musora-content-services 2.158.2 → 2.159.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (54) hide show
  1. package/.claude/settings.local.json +12 -0
  2. package/.github/workflows/automated-testing.yml +21 -1
  3. package/CHANGELOG.md +15 -0
  4. package/README.md +21 -2
  5. package/jest.config.js +1 -4
  6. package/jest.integration.config.js +6 -0
  7. package/jest.live.config.js +1 -5
  8. package/package.json +5 -2
  9. package/src/contentTypeConfig.js +8 -5
  10. package/src/index.d.ts +2 -6
  11. package/src/index.js +2 -6
  12. package/src/services/content-org/learning-paths.ts +44 -39
  13. package/src/services/contentAggregator.js +1 -1
  14. package/src/services/contentProgress.js +216 -207
  15. package/src/services/offline/progress.ts +107 -27
  16. package/src/services/sanity.js +55 -64
  17. package/src/services/sync/models/ContentProgress.ts +50 -34
  18. package/src/services/sync/repositories/content-progress.ts +105 -92
  19. package/test/{unit → integration}/awards/award-exclusion-handling.test.ts +2 -2
  20. package/test/integration/content-progress/__mocks__/mocks.ts +104 -0
  21. package/test/integration/content-progress/contentProgress.test.ts +335 -0
  22. package/test/integration/content-progress/e2eOfflineProgress.test.ts +352 -0
  23. package/test/integration/content-progress/e2eProgress.test.ts +612 -0
  24. package/test/integration/content-progress/getters.test.ts +334 -0
  25. package/test/integration/content-progress/helpers.test.ts +263 -0
  26. package/test/integration/content-progress/offlineContentProgress.test.ts +226 -0
  27. package/test/integration/forums.test.ts +209 -0
  28. package/test/integration/initializeTestDB.ts +80 -0
  29. package/test/{unit → integration}/sync/fetch.test.ts +1 -1
  30. package/test/{unit → integration}/sync/repositories/content-likes.test.ts +1 -1
  31. package/test/{unit → integration}/sync/repositories/practices.test.ts +1 -1
  32. package/test/{unit → integration}/sync/repositories/progress.test.ts +1 -1
  33. package/test/{unit → integration}/sync/repositories/user-award-progress.test.ts +1 -1
  34. package/test/{unit → integration}/sync/store/cross-user-protection.test.ts +2 -2
  35. package/test/{unit → integration}/sync/store/store-idb.test.ts +2 -2
  36. package/test/{unit → integration}/sync/store/store.test.ts +2 -2
  37. package/test/unit/content-progress/bubbleTrickle.test.ts +322 -0
  38. package/test/unit/content-progress/helpers.test.ts +329 -0
  39. package/test/unit/content-progress/navigateTo.test.ts +381 -0
  40. package/test/unit/contentMetaData.test.ts +58 -0
  41. package/tools/generate-index.cjs +6 -3
  42. package/test/SKIPPED_TESTS.md +0 -151
  43. package/test/integration/content.test.js +0 -107
  44. package/test/integration/contentProgress.test.js +0 -73
  45. package/test/integration/forum.test.js +0 -16
  46. package/test/integration/sanityQueryService.test.js +0 -681
  47. package/test/unit/contentProgress.test.ts +0 -81
  48. /package/test/{unit → integration}/awards/internal/image-utils.test.ts +0 -0
  49. /package/test/{unit → integration}/infrastructure/FetchRequestExecutor.test.ts +0 -0
  50. /package/test/{unit → integration}/notifications.test.ts +0 -0
  51. /package/test/{unit → integration}/sync/adapters/idb-errors.test.ts +0 -0
  52. /package/test/{unit → integration}/sync/adapters/sqlite-errors.test.ts +0 -0
  53. /package/test/{unit → integration}/sync/repositories/user-award-progress.static.test.ts +0 -0
  54. /package/test/{unit → integration}/userActivity.test.ts +0 -0
@@ -1,107 +0,0 @@
1
- import { initializeTestService } from '../initializeTests.js'
2
- import {getContentRows, getNewAndUpcoming, getScheduleContentRows, getTabResults} from '../../src/index.js'
3
-
4
- // Mock fetchContentProgress before other modules load
5
- jest.mock('../../src/services/railcontent.js', () => ({
6
- ...jest.requireActual('../../src/services/railcontent.js'),
7
- fetchContentProgress: jest.fn().mockResolvedValue({ version: 1, data: {} })
8
- }))
9
-
10
- const railContentModule = require('../../src/services/railcontent.js')
11
-
12
- describe('content', function () {
13
- beforeAll(async () => {
14
- await initializeTestService(true)
15
- })
16
-
17
- test('getTabResults-Singles', async () => {
18
- const results = await getTabResults('drumeo','lessons','Individuals', {selectedFilters:['difficulty,All','difficulty,Beginner'], sort:'-published_on'})
19
- console.log(results)
20
- expect(results.type).toBeDefined()
21
- expect(results.type).toBe('catalog')
22
- expect(results.data).toBeDefined()
23
- expect(results.meta).toBeDefined()
24
- expect(results.meta.filters).toBeDefined()
25
- expect(results.meta.sort).toBeDefined()
26
- })
27
-
28
- test('getTabResults-Courses', async () => {
29
- const results = await getTabResults('pianote','lessons','Collections', {selectedFilters:['difficulty,Expert'], sort:'slug'})
30
- console.log(results)
31
- expect(results.type).toBeDefined()
32
- expect(results.type).toBe('catalog')
33
- expect(results.data).toBeDefined()
34
- expect(results.meta).toBeDefined()
35
- expect(results.meta.filters).toBeDefined()
36
- expect(results.meta.sort).toBeDefined()
37
- })
38
-
39
- test('getTabResults-Filters', async () => {
40
- const results = await getTabResults('pianote','lessons','Explore All', {selectedFilters:['difficulty,Expert'], sort:'slug'})
41
- console.log(results)
42
- expect(results.type).toBeDefined()
43
- expect(results.data).toBeDefined()
44
- expect(results.meta).toBeDefined()
45
- expect(results.meta.filters).toBeDefined()
46
- expect(results.meta.sort).toBeDefined()
47
- })
48
-
49
- test('getTabResults-Type-Filter', async () => {
50
- const results = await getTabResults('drumeo','lessons','Explore All', {selectedFilters:['type,Courses', 'type,Documentaries'], sort:'slug'})
51
- console.log(results)
52
- expect(results.type).toBeDefined()
53
- expect(results.data).toBeDefined()
54
- expect(results.meta).toBeDefined()
55
- expect(results.meta.filters).toBeDefined()
56
- expect(results.meta.sort).toBeDefined()
57
- })
58
-
59
- test('getTabResults-Type-Explore-All', async () => {
60
- const results = await getTabResults('drumeo','lessons','Explore All', {selectedFilters:[], sort:'slug'})
61
- console.log(results)
62
- expect(results.type).toBeDefined()
63
- expect(results.data).toBeDefined()
64
- expect(results.meta).toBeDefined()
65
- expect(results.meta.filters).toBeDefined()
66
- expect(results.meta.sort).toBeDefined()
67
- })
68
-
69
- test('getContentRows', async () => {
70
- const results = await getContentRows('drumeo', 'lessons', 'Your-Daily-Warmup')
71
- console.log(results)
72
- })
73
-
74
- // test('getTabResults-Songs-For-You', async () => {
75
- // const results = await getTabResults('drumeo','songs','For You')
76
- // console.log(results)
77
- // expect(results.type).toBeDefined()
78
- // expect(results.type).toBe('sections')
79
- // expect(results.data).toBeDefined()
80
- // expect(results.meta).toBeDefined()
81
- // expect(results.meta.filters).toBeDefined()
82
- // expect(results.meta.sort).toBeDefined()
83
- // })
84
- test('getNewAndUpcoming', async () => {
85
- const results = await getNewAndUpcoming('drumeo')
86
- console.log(results)
87
- //expect(results.data).toBeDefined()
88
- })
89
-
90
- test('getScheduleContentRows', async () => {
91
- const results = await getScheduleContentRows('drumeo')
92
- console.log(results.data[1])
93
- expect(results.type).toBeDefined()
94
- expect(results.type).toBe('sections')
95
- expect(results.data).toBeDefined()
96
- expect(results.meta).toBeDefined()
97
- })
98
-
99
- test('getSpecificScheduleContentRow', async () => {
100
- const results = await getScheduleContentRows('drumeo', 'Leaving-Soon')
101
- console.log(results)
102
- expect(results.type).toBeDefined()
103
- expect(results.type).toBe('catalog')
104
- expect(results.data).toBeDefined()
105
- expect(results.meta).toBeDefined()
106
- })
107
- })
@@ -1,73 +0,0 @@
1
- import { initializeTestService } from '../initializeTests.js'
2
- import {getTabResults} from '../../src/index.js';
3
- import {tutorialsLessonTypes, transcriptionsLessonTypes, playAlongLessonTypes} from "../../src/contentTypeConfig.js";
4
-
5
- let mockProgressRecords = []
6
-
7
- jest.mock('../../src/services/sync/repository-proxy.ts', () => {
8
- const mockFns = {
9
- contentProgress: {
10
- getOneProgressByContentId: jest.fn().mockImplementation((contentId) => {
11
- const record = mockProgressRecords.find(r => r.content_id === contentId)
12
- return Promise.resolve({ data: record || null })
13
- }),
14
- getSomeProgressByContentIds: jest.fn().mockImplementation((contentIds) => {
15
- const records = mockProgressRecords.filter(r => contentIds.includes(r.content_id))
16
- return Promise.resolve({ data: records })
17
- }),
18
- started: jest.fn().mockImplementation((limit, opts) => {
19
- const startedIds = mockProgressRecords
20
- .filter(r => r.state === 'started')
21
- .sort((a, b) => b.updated_at - a.updated_at)
22
- .map(r => r.content_id)
23
- const result = limit ? startedIds.slice(0, limit) : startedIds
24
- return Promise.resolve(opts?.onlyIds !== false ? result : result.map(id => ({ content_id: id })))
25
- }),
26
- startedOrCompleted: jest.fn().mockImplementation(() => {
27
- const records = mockProgressRecords
28
- .filter(r => r.state === 'started' || r.state === 'completed')
29
- .sort((a, b) => b.updated_at - a.updated_at)
30
- return Promise.resolve({ data: records })
31
- }),
32
- },
33
- practices: {
34
- queryAll: jest.fn().mockResolvedValue({ data: [] }),
35
- getAll: jest.fn().mockResolvedValue({ data: [] }),
36
- },
37
- }
38
- return { default: mockFns, ...mockFns }
39
- })
40
-
41
- describe('contentProgressDataContext', function () {
42
- beforeEach(() => {
43
- initializeTestService()
44
- mockProgressRecords = [
45
- { content_id: 234191, state: 'started', progress_percent: 6, updated_at: 1731108082, last_interacted_a_la_carte: 1731108082 },
46
- { content_id: 233955, state: 'started', progress_percent: 1, updated_at: 1731108083 },
47
- { content_id: 259426, state: 'completed', progress_percent: 100, updated_at: 1731108085 },
48
- { content_id: 190417, state: 'started', progress_percent: 6, updated_at: 1731108082 },
49
- { content_id: 407665, state: 'started', progress_percent: 6, updated_at: 1740120139 },
50
- { content_id: 412986, state: 'completed', progress_percent: 100, updated_at: 1731108085 },
51
- ]
52
- })
53
-
54
- test.skip('get-Songs-Tutorials', async () => {
55
- const result = await getTabResults('pianote', 'songs', 'Tutorials')
56
- expect(result.type).toStrictEqual('catalog')
57
- expect(result.data).toBeDefined()
58
- expect(tutorialsLessonTypes).toContain(result.data[0].type)
59
- })
60
-
61
- test.skip('get-Songs-Transcriptions', async () => {
62
- const result = await getTabResults('pianote', 'songs', 'Transcriptions')
63
- expect(result.type).toStrictEqual('catalog')
64
- expect(result.data).toBeDefined()
65
- expect(transcriptionsLessonTypes).toContain(result.data[0].type)
66
- })
67
-
68
- test.skip('get-Songs-Play-Alongs', async () => {
69
- const result = await getTabResults('drumeo', 'songs', 'Play-Alongs', { selectedFilters: ['difficulty,Expert'] })
70
- expect(playAlongLessonTypes).toContain(result.data[0].type)
71
- expect(result.data[0].difficulty_string).toStrictEqual('Expert')
72
- })
73
- })
@@ -1,16 +0,0 @@
1
- import { initializeTestService } from '../initializeTests.js'
2
- import {getActiveDiscussions} from "../../src/services/forums/forums.ts";
3
-
4
- describe('forum', function () {
5
- beforeEach(() => {
6
- initializeTestService()
7
- })
8
-
9
- test('getActiveDiscussions', async () => {
10
- const results = await getActiveDiscussions('drumeo')
11
- console.log(results)
12
- expect(results.data).toBeDefined()
13
- expect(results.meta).toBeDefined()
14
- })
15
-
16
- })