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.
- package/.claude/settings.local.json +12 -0
- package/.github/workflows/automated-testing.yml +21 -1
- package/CHANGELOG.md +15 -0
- package/README.md +21 -2
- package/jest.config.js +1 -4
- package/jest.integration.config.js +6 -0
- package/jest.live.config.js +1 -5
- package/package.json +5 -2
- package/src/contentTypeConfig.js +8 -5
- package/src/index.d.ts +2 -6
- package/src/index.js +2 -6
- package/src/services/content-org/learning-paths.ts +44 -39
- package/src/services/contentAggregator.js +1 -1
- package/src/services/contentProgress.js +216 -207
- package/src/services/offline/progress.ts +107 -27
- package/src/services/sanity.js +55 -64
- package/src/services/sync/models/ContentProgress.ts +50 -34
- package/src/services/sync/repositories/content-progress.ts +105 -92
- package/test/{unit → integration}/awards/award-exclusion-handling.test.ts +2 -2
- package/test/integration/content-progress/__mocks__/mocks.ts +104 -0
- package/test/integration/content-progress/contentProgress.test.ts +335 -0
- package/test/integration/content-progress/e2eOfflineProgress.test.ts +352 -0
- package/test/integration/content-progress/e2eProgress.test.ts +612 -0
- package/test/integration/content-progress/getters.test.ts +334 -0
- package/test/integration/content-progress/helpers.test.ts +263 -0
- package/test/integration/content-progress/offlineContentProgress.test.ts +226 -0
- package/test/integration/forums.test.ts +209 -0
- package/test/integration/initializeTestDB.ts +80 -0
- package/test/{unit → integration}/sync/fetch.test.ts +1 -1
- package/test/{unit → integration}/sync/repositories/content-likes.test.ts +1 -1
- package/test/{unit → integration}/sync/repositories/practices.test.ts +1 -1
- package/test/{unit → integration}/sync/repositories/progress.test.ts +1 -1
- package/test/{unit → integration}/sync/repositories/user-award-progress.test.ts +1 -1
- package/test/{unit → integration}/sync/store/cross-user-protection.test.ts +2 -2
- package/test/{unit → integration}/sync/store/store-idb.test.ts +2 -2
- package/test/{unit → integration}/sync/store/store.test.ts +2 -2
- package/test/unit/content-progress/bubbleTrickle.test.ts +322 -0
- package/test/unit/content-progress/helpers.test.ts +329 -0
- package/test/unit/content-progress/navigateTo.test.ts +381 -0
- package/test/unit/contentMetaData.test.ts +58 -0
- package/tools/generate-index.cjs +6 -3
- package/test/SKIPPED_TESTS.md +0 -151
- package/test/integration/content.test.js +0 -107
- package/test/integration/contentProgress.test.js +0 -73
- package/test/integration/forum.test.js +0 -16
- package/test/integration/sanityQueryService.test.js +0 -681
- package/test/unit/contentProgress.test.ts +0 -81
- /package/test/{unit → integration}/awards/internal/image-utils.test.ts +0 -0
- /package/test/{unit → integration}/infrastructure/FetchRequestExecutor.test.ts +0 -0
- /package/test/{unit → integration}/notifications.test.ts +0 -0
- /package/test/{unit → integration}/sync/adapters/idb-errors.test.ts +0 -0
- /package/test/{unit → integration}/sync/adapters/sqlite-errors.test.ts +0 -0
- /package/test/{unit → integration}/sync/repositories/user-award-progress.static.test.ts +0 -0
- /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
|
-
})
|