steelseries-sonar-sdk 0.1.1

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 (118) hide show
  1. package/.gitattributes +8 -0
  2. package/.github/workflows/tests.yml +18 -0
  3. package/.vscode/extensions.json +8 -0
  4. package/.vscode/settings.json +33 -0
  5. package/LICENSE +21 -0
  6. package/README.md +121 -0
  7. package/_.todo +4 -0
  8. package/biome.json +62 -0
  9. package/http/sonar.http +104 -0
  10. package/package.json +31 -0
  11. package/src/consts/fetch-options-put.ts +3 -0
  12. package/src/enums.ts +88 -0
  13. package/src/exceptions.ts +45 -0
  14. package/src/functions/audio/get-audio-data-classic.ts +31 -0
  15. package/src/functions/audio/get-audio-data-stream.ts +37 -0
  16. package/src/functions/audio/get-audio-mode.ts +11 -0
  17. package/src/functions/audio/set-audio-mode.ts +12 -0
  18. package/src/functions/audio/set-channel-mute-classic.ts +35 -0
  19. package/src/functions/audio/set-channel-mute-streamer.ts +41 -0
  20. package/src/functions/audio/set-channel-volume-classic.ts +37 -0
  21. package/src/functions/audio/set-channel-volume-streamer.ts +43 -0
  22. package/src/functions/chatmix/get-chat-mix-state.ts +20 -0
  23. package/src/functions/chatmix/set-chat-mix-balance.ts +22 -0
  24. package/src/functions/converters/convert-channel-to-api.ts +19 -0
  25. package/src/functions/converters/convert-chat-mix-balance-to-api.ts +6 -0
  26. package/src/functions/converters/convert-chat-mix-balance-to-user.ts +7 -0
  27. package/src/functions/converters/convert-profile-channel-to-api.ts +19 -0
  28. package/src/functions/converters/convert-profile-channel-to-user.ts +19 -0
  29. package/src/functions/converters/convert-volume-to-api.ts +5 -0
  30. package/src/functions/converters/convert-volume-to-user.ts +6 -0
  31. package/src/functions/devices/get-audio-devices.ts +40 -0
  32. package/src/functions/devices/set-audio-device.ts +19 -0
  33. package/src/functions/endpoint/get-app-endpoint.ts +73 -0
  34. package/src/functions/endpoint/get-sonar-endpoint-cached.ts +28 -0
  35. package/src/functions/endpoint/get-sonar-endpoint.ts +88 -0
  36. package/src/functions/profile/get-channel-profiles.ts +26 -0
  37. package/src/functions/profile/get-selected-profiles.ts +17 -0
  38. package/src/functions/profile/set-selected-profile.ts +17 -0
  39. package/src/index.ts +41 -0
  40. package/src/sonar/models/audio-settings/chatmix-data.ts +6 -0
  41. package/src/sonar/models/audio-settings/enums/audio-mode.ts +4 -0
  42. package/src/sonar/models/audio-settings/enums/chat-mix-state.ts +6 -0
  43. package/src/sonar/models/audio-settings/enums/device-data-flow.ts +5 -0
  44. package/src/sonar/models/audio-settings/enums/device-default-roles.ts +6 -0
  45. package/src/sonar/models/audio-settings/enums/device-role.ts +9 -0
  46. package/src/sonar/models/audio-settings/enums/device-state.ts +8 -0
  47. package/src/sonar/models/audio-settings/enums/sonar-channel.ts +8 -0
  48. package/src/sonar/models/audio-settings/enums/streaming-path.ts +4 -0
  49. package/src/sonar/models/audio-settings/volume-info-classic.ts +4 -0
  50. package/src/sonar/models/audio-settings/volume-info-streamer.ts +10 -0
  51. package/src/sonar/models/audio-settings/volume-settings-classic.ts +13 -0
  52. package/src/sonar/models/audio-settings/volume-settings-streamer.ts +13 -0
  53. package/src/sonar/models/config/config.ts +9 -0
  54. package/src/sonar/models/config/enums/VirtualAudioDevice.ts +7 -0
  55. package/src/sonar/models/devices/audio-device.ts +17 -0
  56. package/src/sonar/models/devices/changed-device.ts +7 -0
  57. package/src/sonar/requests/audio-devices/change-audio-device.ts +37 -0
  58. package/src/sonar/requests/audio-devices/request-audio-devices.ts +42 -0
  59. package/src/sonar/requests/chatmix/change-chat-mix-balance.ts +25 -0
  60. package/src/sonar/requests/chatmix/request-chat-mix-state.ts +24 -0
  61. package/src/sonar/requests/mode/change-audio-mode.ts +24 -0
  62. package/src/sonar/requests/mode/request-audio-mode.ts +24 -0
  63. package/src/sonar/requests/profiles/change-selected-config.ts +25 -0
  64. package/src/sonar/requests/profiles/request-configs.ts +24 -0
  65. package/src/sonar/requests/profiles/request-selected-configs.ts +22 -0
  66. package/src/sonar/requests/volume-settings/change-volume-level-classic.ts +27 -0
  67. package/src/sonar/requests/volume-settings/change-volume-level-streamer.ts +29 -0
  68. package/src/sonar/requests/volume-settings/change-volume-mute-classic.ts +27 -0
  69. package/src/sonar/requests/volume-settings/change-volume-mute-streamer.ts +29 -0
  70. package/src/sonar/requests/volume-settings/request-volume-settings-classic.ts +22 -0
  71. package/src/sonar/requests/volume-settings/request-volume-settings-streamer.ts +22 -0
  72. package/src/types/audio-device.ts +7 -0
  73. package/src/types/changed-device.ts +7 -0
  74. package/src/types/channel-volume-classic.ts +4 -0
  75. package/src/types/channel-volume-streamer-path.ts +4 -0
  76. package/src/types/channel-volume-streamer.ts +6 -0
  77. package/src/types/channel-volumes-classic.ts +8 -0
  78. package/src/types/channel-volumes-streamer.ts +8 -0
  79. package/src/types/chat-mix-data.ts +7 -0
  80. package/src/types/profile-option.ts +9 -0
  81. package/tests/e2e/audio/get-audio-data-classic.real.test.ts +11 -0
  82. package/tests/e2e/audio/get-audio-data-stream.real.test.ts +11 -0
  83. package/tests/e2e/audio/get-audio-mode.real.test.ts +12 -0
  84. package/tests/e2e/audio/set-audio-mode.real.test.ts +20 -0
  85. package/tests/e2e/audio/set-channel-mute-classic.real.test.ts +18 -0
  86. package/tests/e2e/audio/set-channel-mute-streamer.real.test.ts +18 -0
  87. package/tests/e2e/audio/set-channel-volume-classic.real.test.ts +17 -0
  88. package/tests/e2e/audio/set-channel-volume-streamer.real.test.ts +22 -0
  89. package/tests/e2e/chatmix/get-chat-mix-state.real.test.ts +25 -0
  90. package/tests/e2e/chatmix/set-chat-mix-balance.real.test.ts +25 -0
  91. package/tests/e2e/devices/get-audio-devices.real.test.ts +29 -0
  92. package/tests/e2e/devices/set-audio-device.real.test.ts +32 -0
  93. package/tests/e2e/enpoint/get-app-endpoint.real.test.ts +19 -0
  94. package/tests/e2e/enpoint/get-sonar-endpoint-cached.real.test.ts +32 -0
  95. package/tests/e2e/enpoint/get-sonar-endpoint.real.test.ts +21 -0
  96. package/tests/e2e/profiles/get-channel-profiles.real.test.ts +21 -0
  97. package/tests/e2e/profiles/get-selected-profiles.real.test.ts +11 -0
  98. package/tests/e2e/profiles/set-selected-profile.real.test.ts +17 -0
  99. package/tests/helpers/get-endpoint-e2e.ts +8 -0
  100. package/tests/unit/audio/get-audio-data-classic.test.ts +57 -0
  101. package/tests/unit/audio/get-audio-data-stream.test.ts +65 -0
  102. package/tests/unit/audio/get-audio-mode.test.ts +58 -0
  103. package/tests/unit/audio/set-audio-mode.test.ts +66 -0
  104. package/tests/unit/audio/set-channel-mute-classic.test.ts +58 -0
  105. package/tests/unit/audio/set-channel-mute-streamer.test.ts +64 -0
  106. package/tests/unit/audio/set-channel-volume-classic.test.ts +58 -0
  107. package/tests/unit/audio/set-channel-volume-streamer.test.ts +64 -0
  108. package/tests/unit/chatmix/get-chat-mix-state.test.ts +70 -0
  109. package/tests/unit/chatmix/set-chat-mix-balance.test.ts +58 -0
  110. package/tests/unit/devices/get-audio-devices.test.ts +59 -0
  111. package/tests/unit/devices/set-audio-devices.test.ts +62 -0
  112. package/tests/unit/endpoint/get-app-endpoint.test.ts +72 -0
  113. package/tests/unit/endpoint/get-sonar-endpoint-cached.test.ts +93 -0
  114. package/tests/unit/endpoint/get-sonar-endpoint.test.ts +140 -0
  115. package/tests/unit/profiles/get-channel-profiles.test.ts +91 -0
  116. package/tests/unit/profiles/get-selected-profiles.test.ts +64 -0
  117. package/tests/unit/profiles/set-selected-profile.test.ts +55 -0
  118. package/tsconfig.json +36 -0
@@ -0,0 +1,58 @@
1
+ import { afterEach, beforeEach, describe, expect, it } from 'bun:test'
2
+ import { AudioChannel } from '@/enums'
3
+ import { SonarRequestException } from '@/exceptions'
4
+ import { setChannelMuteClassic } from '@/functions/audio/set-channel-mute-classic'
5
+
6
+ let originalFetch: typeof fetch
7
+
8
+ const request = () => setChannelMuteClassic('https://localhost', true, AudioChannel.Master)
9
+
10
+ describe('setChannelMuteClassic', () => {
11
+ beforeEach(() => {
12
+ originalFetch = globalThis.fetch
13
+ })
14
+
15
+ afterEach(() => {
16
+ globalThis.fetch = originalFetch
17
+ })
18
+
19
+ it('throws SonarException when response not 200', async () => {
20
+ globalThis.fetch = (async () =>
21
+ ({
22
+ ok: false,
23
+ json: async () => 'Some error occurred'
24
+ }) as Response) as unknown as typeof fetch
25
+
26
+ expect(request()).rejects.toThrow(SonarRequestException)
27
+ })
28
+
29
+ it('throws SonarException when response 200 but not data', async () => {
30
+ globalThis.fetch = (async () =>
31
+ ({
32
+ ok: true,
33
+ json: async () => ({})
34
+ }) as Response) as unknown as typeof fetch
35
+
36
+ expect(request()).rejects.toThrow(SonarRequestException)
37
+ })
38
+
39
+ it('return data when response 200', async () => {
40
+ globalThis.fetch = (async () =>
41
+ ({
42
+ ok: true,
43
+ json: async () => ({
44
+ masters: {
45
+ classic: {
46
+ volume: 0.49,
47
+ muted: false
48
+ }
49
+ },
50
+ devices: {}
51
+ })
52
+ }) as Response) as unknown as typeof fetch
53
+
54
+ const response = await request()
55
+ expect(response.volume).toBe(49)
56
+ expect(response.isMuted).toBe(false)
57
+ })
58
+ })
@@ -0,0 +1,64 @@
1
+ import { afterEach, beforeEach, describe, expect, it } from 'bun:test'
2
+ import { AudioChannel, StreamerPath } from '@/enums'
3
+ import { SonarRequestException } from '@/exceptions'
4
+ import { setChannelMuteStreamer } from '@/functions/audio/set-channel-mute-streamer'
5
+
6
+ let originalFetch: typeof fetch
7
+
8
+ const request = () => setChannelMuteStreamer('https://localhost', false, AudioChannel.Master, StreamerPath.Streaming)
9
+
10
+ describe('setChannelMuteStreamer', () => {
11
+ beforeEach(() => {
12
+ originalFetch = globalThis.fetch
13
+ })
14
+
15
+ afterEach(() => {
16
+ globalThis.fetch = originalFetch
17
+ })
18
+
19
+ it('throws SonarException when response not 200', async () => {
20
+ globalThis.fetch = (async () =>
21
+ ({
22
+ ok: false,
23
+ json: async () => 'Some error occurred'
24
+ }) as Response) as unknown as typeof fetch
25
+
26
+ expect(request()).rejects.toThrow(SonarRequestException)
27
+ })
28
+
29
+ it('throws SonarException when response 200 but not data', async () => {
30
+ globalThis.fetch = (async () =>
31
+ ({
32
+ ok: true,
33
+ json: async () => ({})
34
+ }) as Response) as unknown as typeof fetch
35
+
36
+ expect(request()).rejects.toThrow(SonarRequestException)
37
+ })
38
+
39
+ it('return data when response 200', async () => {
40
+ globalThis.fetch = (async () =>
41
+ ({
42
+ ok: true,
43
+ json: async () => ({
44
+ masters: {
45
+ stream: {
46
+ streaming: {
47
+ volume: 0.49,
48
+ muted: true
49
+ },
50
+ monitoring: {
51
+ volume: 0.48,
52
+ muted: false
53
+ }
54
+ }
55
+ },
56
+ devices: {}
57
+ })
58
+ }) as Response) as unknown as typeof fetch
59
+
60
+ const response = await request()
61
+ expect(response.volume).toBe(49)
62
+ expect(response.isMuted).toBe(true)
63
+ })
64
+ })
@@ -0,0 +1,58 @@
1
+ import { afterEach, beforeEach, describe, expect, it } from 'bun:test'
2
+ import { AudioChannel } from '@/enums'
3
+ import { SonarRequestException } from '@/exceptions'
4
+ import { setChannelVolumeClassic } from '@/functions/audio/set-channel-volume-classic'
5
+
6
+ let originalFetch: typeof fetch
7
+
8
+ const request = () => setChannelVolumeClassic('https://localhost', 75, AudioChannel.Master)
9
+
10
+ describe('setChannelVolumeClassic', () => {
11
+ beforeEach(() => {
12
+ originalFetch = globalThis.fetch
13
+ })
14
+
15
+ afterEach(() => {
16
+ globalThis.fetch = originalFetch
17
+ })
18
+
19
+ it('throws SonarException when response not 200', async () => {
20
+ globalThis.fetch = (async () =>
21
+ ({
22
+ ok: false,
23
+ json: async () => 'Some error occurred'
24
+ }) as Response) as unknown as typeof fetch
25
+
26
+ expect(request()).rejects.toThrow(SonarRequestException)
27
+ })
28
+
29
+ it('throws SonarException when response 200 but not data', async () => {
30
+ globalThis.fetch = (async () =>
31
+ ({
32
+ ok: true,
33
+ json: async () => ({})
34
+ }) as Response) as unknown as typeof fetch
35
+
36
+ expect(request()).rejects.toThrow(SonarRequestException)
37
+ })
38
+
39
+ it('return data when response 200', async () => {
40
+ globalThis.fetch = (async () =>
41
+ ({
42
+ ok: true,
43
+ json: async () => ({
44
+ masters: {
45
+ classic: {
46
+ volume: 0.49,
47
+ muted: false
48
+ }
49
+ },
50
+ devices: {}
51
+ })
52
+ }) as Response) as unknown as typeof fetch
53
+
54
+ const response = await request()
55
+ expect(response.volume).toBe(49)
56
+ expect(response.isMuted).toBe(false)
57
+ })
58
+ })
@@ -0,0 +1,64 @@
1
+ import { afterEach, beforeEach, describe, expect, it } from 'bun:test'
2
+ import { AudioChannel, StreamerPath } from '@/enums'
3
+ import { SonarRequestException } from '@/exceptions'
4
+ import { setChannelVolumeStreamer } from '@/functions/audio/set-channel-volume-streamer'
5
+
6
+ let originalFetch: typeof fetch
7
+
8
+ const request = () => setChannelVolumeStreamer('https://localhost', 75, AudioChannel.Master, StreamerPath.Streaming)
9
+
10
+ describe('setChannelVolumeStreamer', () => {
11
+ beforeEach(() => {
12
+ originalFetch = globalThis.fetch
13
+ })
14
+
15
+ afterEach(() => {
16
+ globalThis.fetch = originalFetch
17
+ })
18
+
19
+ it('throws SonarException when response not 200', async () => {
20
+ globalThis.fetch = (async () =>
21
+ ({
22
+ ok: false,
23
+ json: async () => 'Some error occurred'
24
+ }) as Response) as unknown as typeof fetch
25
+
26
+ expect(request()).rejects.toThrow(SonarRequestException)
27
+ })
28
+
29
+ it('throws SonarException when response 200 but not data', async () => {
30
+ globalThis.fetch = (async () =>
31
+ ({
32
+ ok: true,
33
+ json: async () => ({})
34
+ }) as Response) as unknown as typeof fetch
35
+
36
+ expect(request()).rejects.toThrow(SonarRequestException)
37
+ })
38
+
39
+ it('return data when response 200', async () => {
40
+ globalThis.fetch = (async () =>
41
+ ({
42
+ ok: true,
43
+ json: async () => ({
44
+ masters: {
45
+ stream: {
46
+ streaming: {
47
+ volume: 0.49,
48
+ muted: true
49
+ },
50
+ monitoring: {
51
+ volume: 0.48,
52
+ muted: false
53
+ }
54
+ }
55
+ },
56
+ devices: {}
57
+ })
58
+ }) as Response) as unknown as typeof fetch
59
+
60
+ const response = await request()
61
+ expect(response.volume).toBe(49)
62
+ expect(response.isMuted).toBe(true)
63
+ })
64
+ })
@@ -0,0 +1,70 @@
1
+ import { afterEach, beforeEach, describe, expect, it } from 'bun:test'
2
+ import { ChatMixState } from '@/enums'
3
+ import { SonarRequestException } from '@/exceptions'
4
+ import { getChatMixState } from '@/functions/chatmix/get-chat-mix-state'
5
+
6
+ let originalFetch: typeof fetch
7
+
8
+ const request = () => getChatMixState('http://localhost')
9
+
10
+ describe('getChatMixState', () => {
11
+ beforeEach(() => {
12
+ originalFetch = globalThis.fetch
13
+ })
14
+
15
+ afterEach(() => {
16
+ globalThis.fetch = originalFetch
17
+ })
18
+
19
+ it('throws SonarException when response not 200', async () => {
20
+ globalThis.fetch = (async () =>
21
+ ({
22
+ ok: false,
23
+ text: async () => 'Some error occurred'
24
+ }) as Response) as unknown as typeof fetch
25
+
26
+ expect(request()).rejects.toThrow(SonarRequestException)
27
+ })
28
+
29
+ it('throws SonarException when response 200 but not data', async () => {
30
+ globalThis.fetch = (async () =>
31
+ ({
32
+ ok: true,
33
+ json: async () => ({})
34
+ }) as Response) as unknown as typeof fetch
35
+
36
+ expect(request()).rejects.toThrow(SonarRequestException)
37
+ })
38
+
39
+ it('return data when classic and response 200', async () => {
40
+ globalThis.fetch = (async () =>
41
+ ({
42
+ ok: true,
43
+ json: async () => ({
44
+ balance: 0.5,
45
+ state: ChatMixState.Enabled
46
+ })
47
+ }) as Response) as unknown as typeof fetch
48
+
49
+ const response = await request()
50
+ expect(response.chatBalance).toBe(75)
51
+ expect(response.state).toBe(ChatMixState.Enabled)
52
+ expect(response.isEnabled).toBeTrue()
53
+ })
54
+
55
+ it('return data when streamer and response 200', async () => {
56
+ globalThis.fetch = (async () =>
57
+ ({
58
+ ok: true,
59
+ json: async () => ({
60
+ balance: 0.5,
61
+ state: ChatMixState.FiniteWheel
62
+ })
63
+ }) as Response) as unknown as typeof fetch
64
+
65
+ const response = await request()
66
+ expect(response.chatBalance).toBe(75)
67
+ expect(response.state).toBe(ChatMixState.FiniteWheel)
68
+ expect(response.isEnabled).toBeFalse()
69
+ })
70
+ })
@@ -0,0 +1,58 @@
1
+ import { afterEach, beforeEach, describe, expect, it } from 'bun:test'
2
+ import { ChatMixState } from '@/enums'
3
+ import { SonarRequestException } from '@/exceptions'
4
+ import { setChatMixBalance } from '@/functions/chatmix/set-chat-mix-balance'
5
+
6
+ let originalFetch: typeof fetch
7
+
8
+ const request = (chatBalance: number) => setChatMixBalance('http://localhost', chatBalance)
9
+
10
+ describe('setChatBalance', () => {
11
+ beforeEach(() => {
12
+ originalFetch = globalThis.fetch
13
+ })
14
+
15
+ afterEach(() => {
16
+ globalThis.fetch = originalFetch
17
+ })
18
+
19
+ it('throws SonarException when response not 200', async () => {
20
+ globalThis.fetch = (async () =>
21
+ ({
22
+ ok: false,
23
+ json: async () => ({
24
+ message: 'Some error occurred'
25
+ })
26
+ }) as Response) as unknown as typeof fetch
27
+
28
+ expect(request(0)).rejects.toThrow(SonarRequestException)
29
+ })
30
+
31
+ it('throws SonarException when response 200 but not data', async () => {
32
+ globalThis.fetch = (async () =>
33
+ ({
34
+ ok: true,
35
+ json: async () => ({
36
+ message: 'Some error occurred'
37
+ })
38
+ }) as Response) as unknown as typeof fetch
39
+
40
+ expect(request(0)).rejects.toThrow(SonarRequestException)
41
+ })
42
+
43
+ it('return data when response 200', async () => {
44
+ globalThis.fetch = (async () =>
45
+ ({
46
+ ok: true,
47
+ json: async () => ({
48
+ balance: 0.5,
49
+ state: ChatMixState.Enabled
50
+ })
51
+ }) as Response) as unknown as typeof fetch
52
+
53
+ const response = await request(0.5)
54
+ expect(response.chatBalance).toBe(75)
55
+ expect(response.state).toBe(ChatMixState.Enabled)
56
+ expect(response.isEnabled).toBeTrue()
57
+ })
58
+ })
@@ -0,0 +1,59 @@
1
+ import { afterEach, beforeEach, describe, expect, it } from 'bun:test'
2
+ import { SonarRequestException } from '@/exceptions'
3
+ import { getAudioDevices } from '@/functions/devices/get-audio-devices'
4
+
5
+ let originalFetch: typeof fetch
6
+
7
+ const request = () => getAudioDevices('http://localhost')
8
+
9
+ describe('getAudioDevices', () => {
10
+ beforeEach(() => {
11
+ originalFetch = globalThis.fetch
12
+ })
13
+
14
+ afterEach(() => {
15
+ globalThis.fetch = originalFetch
16
+ })
17
+
18
+ it('throws SonarException when response not 200', async () => {
19
+ globalThis.fetch = (async () =>
20
+ ({
21
+ ok: false,
22
+ text: async () => 'Some error occurred'
23
+ }) as Response) as unknown as typeof fetch
24
+
25
+ expect(request()).rejects.toThrow(SonarRequestException)
26
+ })
27
+
28
+ it('throws SonarException when response 200 but not data', async () => {
29
+ globalThis.fetch = (async () =>
30
+ ({
31
+ ok: true,
32
+ json: async () => ({})
33
+ }) as Response) as unknown as typeof fetch
34
+
35
+ expect(request()).rejects.toThrow(SonarRequestException)
36
+ })
37
+
38
+ it('return data when response 200', async () => {
39
+ globalThis.fetch = (async () =>
40
+ ({
41
+ ok: true,
42
+ json: async () => [
43
+ {
44
+ friendlyName: 'Speakers',
45
+ id: 'speaker',
46
+ dataFlow: 'render'
47
+ },
48
+ {
49
+ friendlyName: 'Microphone',
50
+ id: 'mic',
51
+ dataFlow: 'capture'
52
+ }
53
+ ]
54
+ }) as Response) as unknown as typeof fetch
55
+
56
+ const response = await request()
57
+ expect(response.length).toBe(2)
58
+ })
59
+ })
@@ -0,0 +1,62 @@
1
+ import { afterEach, beforeEach, describe, expect, it } from 'bun:test'
2
+ import { DeviceChannel } from '@/enums'
3
+ import { SonarRequestException } from '@/exceptions'
4
+ import { setAudioDevice } from '@/functions/devices/set-audio-device'
5
+
6
+ let originalFetch: typeof fetch
7
+
8
+ const request = () => setAudioDevice('http://localhost', DeviceChannel.Output, 'game_ID')
9
+
10
+ describe('setAudioDevices', () => {
11
+ beforeEach(() => {
12
+ originalFetch = globalThis.fetch
13
+ })
14
+
15
+ afterEach(() => {
16
+ globalThis.fetch = originalFetch
17
+ })
18
+
19
+ it('throws SonarException when response not 200', async () => {
20
+ globalThis.fetch = (async () =>
21
+ ({
22
+ ok: false,
23
+ text: async () => 'Some error occurred'
24
+ }) as Response) as unknown as typeof fetch
25
+
26
+ expect(request()).rejects.toThrow(SonarRequestException)
27
+ })
28
+
29
+ it('throws SonarException when response 200 but not data', async () => {
30
+ globalThis.fetch = (async () =>
31
+ ({
32
+ ok: true,
33
+ json: async () => ({})
34
+ }) as Response) as unknown as typeof fetch
35
+
36
+ expect(request()).rejects.toThrow(SonarRequestException)
37
+ })
38
+
39
+ it('return data when response 200', async () => {
40
+ globalThis.fetch = (async () =>
41
+ ({
42
+ ok: true,
43
+ json: async () => [
44
+ {
45
+ id: 'game',
46
+ deviceId: 'game_ID',
47
+ isRunning: true
48
+ },
49
+ {
50
+ id: 'aux',
51
+ deviceId: 'game_ID',
52
+ isRunning: false
53
+ }
54
+ ]
55
+ }) as Response) as unknown as typeof fetch
56
+
57
+ const response = await request()
58
+ expect(response.length).toBe(2)
59
+ expect(response[0]?.deviceChannel).toBe(DeviceChannel.Game)
60
+ expect(response[1]?.deviceChannel).toBe(DeviceChannel.Aux)
61
+ })
62
+ })
@@ -0,0 +1,72 @@
1
+ import { describe, expect, it, mock } from 'bun:test'
2
+ import { InitializeErrorReason, SonarInitializationException } from '../../../src/exceptions'
3
+ import { getAppEndpoint } from '../../../src/functions/endpoint/get-app-endpoint'
4
+
5
+ describe('getAppEndpoint', () => {
6
+ it('throws Unsupported Exception on unsupported OS', async () => {
7
+ mock.module('node:os', () => ({
8
+ platform: () => 'linux'
9
+ }))
10
+ const rejects = expect(getAppEndpoint()).rejects
11
+ rejects.toThrow(SonarInitializationException)
12
+ rejects.toMatchObject({ reason: InitializeErrorReason.OSUnsupported })
13
+ })
14
+
15
+ it('throws BadConfig Exception when appData file does not exist', async () => {
16
+ mock.module('node:os', () => ({
17
+ platform: () => 'win32'
18
+ }))
19
+ mock.module('node:fs', () => ({
20
+ promises: {
21
+ readFile: async () => {
22
+ throw new Error('File not found')
23
+ }
24
+ }
25
+ }))
26
+ const rejects = expect(getAppEndpoint()).rejects
27
+ rejects.toThrow(SonarInitializationException)
28
+ rejects.toMatchObject({ reason: InitializeErrorReason.BadConfig })
29
+ })
30
+
31
+ it('throws BadConfig Exception when appData file is invalid JSON', async () => {
32
+ mock.module('node:os', () => ({
33
+ platform: () => 'win32'
34
+ }))
35
+ mock.module('node:fs', () => ({
36
+ promises: {
37
+ readFile: async () => 'invalid json'
38
+ }
39
+ }))
40
+ const rejects = expect(getAppEndpoint()).rejects
41
+ rejects.toThrow(SonarInitializationException)
42
+ rejects.toMatchObject({ reason: InitializeErrorReason.BadConfig })
43
+ })
44
+
45
+ it('throws BadConfig Exception when ggEncryptedAddress is missing', async () => {
46
+ mock.module('node:os', () => ({
47
+ platform: () => 'win32'
48
+ }))
49
+ mock.module('node:fs', () => ({
50
+ promises: {
51
+ readFile: async () => JSON.stringify({})
52
+ }
53
+ }))
54
+ const rejects = expect(getAppEndpoint()).rejects
55
+ rejects.toThrow(SonarInitializationException)
56
+ rejects.toMatchObject({ reason: InitializeErrorReason.BadConfig })
57
+ })
58
+
59
+ it('returns ggEncryptedAddress when file is valid', async () => {
60
+ const mockAddress = 'encrypted-address-123'
61
+ const expectedUrl = `https://${mockAddress}`
62
+ mock.module('node:os', () => ({
63
+ platform: () => 'win32'
64
+ }))
65
+ mock.module('node:fs', () => ({
66
+ promises: {
67
+ readFile: async () => JSON.stringify({ ggEncryptedAddress: mockAddress })
68
+ }
69
+ }))
70
+ expect(getAppEndpoint()).resolves.toBe(expectedUrl)
71
+ })
72
+ })
@@ -0,0 +1,93 @@
1
+ import { afterEach, beforeEach, describe, expect, it, setSystemTime } from 'bun:test'
2
+ import { getSonarEndpointCached } from '../../../src/functions/endpoint/get-sonar-endpoint-cached'
3
+
4
+ let originalFetch: typeof fetch
5
+
6
+ describe('getSonarEndpointCached', () => {
7
+ beforeEach(() => {
8
+ originalFetch = globalThis.fetch
9
+ })
10
+
11
+ afterEach(() => {
12
+ globalThis.fetch = originalFetch
13
+ })
14
+
15
+ it('return cached value when within time', async () => {
16
+ const expectedWebServerAddress = 'https://127.0.0.1:1234'
17
+
18
+ globalThis.fetch = (async () =>
19
+ ({
20
+ ok: true,
21
+ json: async () => ({
22
+ subApps: {
23
+ sonar: {
24
+ isEnabled: true,
25
+ isReady: true,
26
+ isRunning: true,
27
+ metadata: { webServerAddress: expectedWebServerAddress }
28
+ }
29
+ }
30
+ })
31
+ }) as Response) as unknown as typeof fetch
32
+
33
+ setSystemTime(1000)
34
+ expect(getSonarEndpointCached('', 1)).resolves.toBe(expectedWebServerAddress)
35
+
36
+ globalThis.fetch = (async () =>
37
+ ({
38
+ ok: true,
39
+ json: async () => ({
40
+ subApps: {
41
+ sonar: {
42
+ isEnabled: true,
43
+ isReady: true,
44
+ isRunning: true,
45
+ metadata: { webServerAddress: 'other' }
46
+ }
47
+ }
48
+ })
49
+ }) as Response) as unknown as typeof fetch
50
+ setSystemTime(500)
51
+ expect(getSonarEndpointCached('', 1)).resolves.toBe(expectedWebServerAddress)
52
+ })
53
+
54
+ it('return new value when cache time expired', async () => {
55
+ const expectedFirstAddress = 'https://127.0.0.1:1234'
56
+ const expectedSecondAddress = 'https://127.0.0.1:4321'
57
+
58
+ globalThis.fetch = (async () =>
59
+ ({
60
+ ok: true,
61
+ json: async () => ({
62
+ subApps: {
63
+ sonar: {
64
+ isEnabled: true,
65
+ isReady: true,
66
+ isRunning: true,
67
+ metadata: { webServerAddress: expectedFirstAddress }
68
+ }
69
+ }
70
+ })
71
+ }) as Response) as unknown as typeof fetch
72
+
73
+ setSystemTime(1000)
74
+ expect(getSonarEndpointCached('', 1)).resolves.toBe(expectedFirstAddress)
75
+
76
+ globalThis.fetch = (async () =>
77
+ ({
78
+ ok: true,
79
+ json: async () => ({
80
+ subApps: {
81
+ sonar: {
82
+ isEnabled: true,
83
+ isReady: true,
84
+ isRunning: true,
85
+ metadata: { webServerAddress: expectedSecondAddress }
86
+ }
87
+ }
88
+ })
89
+ }) as Response) as unknown as typeof fetch
90
+ setSystemTime(2500)
91
+ expect(getSonarEndpointCached('', 1)).resolves.toBe(expectedSecondAddress)
92
+ })
93
+ })