session-wrap-dashboard 3.9.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 (64) hide show
  1. package/.env.example +6 -0
  2. package/coverage/base.css +224 -0
  3. package/coverage/block-navigation.js +87 -0
  4. package/coverage/coverage-final.json +6 -0
  5. package/coverage/favicon.png +0 -0
  6. package/coverage/index.html +146 -0
  7. package/coverage/prettify.css +1 -0
  8. package/coverage/prettify.js +2 -0
  9. package/coverage/sort-arrow-sprite.png +0 -0
  10. package/coverage/sorter.js +210 -0
  11. package/coverage/src/auth.ts.html +307 -0
  12. package/coverage/src/components/Header.tsx.html +262 -0
  13. package/coverage/src/components/Sidebar.tsx.html +262 -0
  14. package/coverage/src/components/index.html +131 -0
  15. package/coverage/src/hooks/index.html +131 -0
  16. package/coverage/src/hooks/useAuth.ts.html +292 -0
  17. package/coverage/src/hooks/useWorkspace.ts.html +475 -0
  18. package/coverage/src/index.html +116 -0
  19. package/index.html +13 -0
  20. package/package.json +49 -0
  21. package/playwright.config.ts +38 -0
  22. package/postcss.config.js +6 -0
  23. package/src/App.tsx +67 -0
  24. package/src/api.ts +130 -0
  25. package/src/auth.ts +74 -0
  26. package/src/components/AgentLeaderboard.tsx +84 -0
  27. package/src/components/AnalyticsDashboard.tsx +223 -0
  28. package/src/components/Header.tsx +62 -0
  29. package/src/components/IntegrationManager.tsx +292 -0
  30. package/src/components/RoleManager.tsx +230 -0
  31. package/src/components/Sidebar.tsx +62 -0
  32. package/src/components/TrendChart.tsx +92 -0
  33. package/src/components/WorkspaceSelector.tsx +157 -0
  34. package/src/components/__tests__/Header.test.tsx +64 -0
  35. package/src/components/__tests__/Sidebar.test.tsx +39 -0
  36. package/src/components/index.ts +8 -0
  37. package/src/hooks/__tests__/useAuth.test.ts +88 -0
  38. package/src/hooks/__tests__/useWorkspace.test.ts +101 -0
  39. package/src/hooks/index.ts +4 -0
  40. package/src/hooks/useAnalytics.ts +76 -0
  41. package/src/hooks/useApi.ts +62 -0
  42. package/src/hooks/useAuth.ts +69 -0
  43. package/src/hooks/useWorkspace.ts +130 -0
  44. package/src/index.css +57 -0
  45. package/src/main.tsx +13 -0
  46. package/src/pages/DashboardPage.tsx +13 -0
  47. package/src/pages/HomePage.tsx +156 -0
  48. package/src/pages/IntegrationsPage.tsx +9 -0
  49. package/src/pages/RolesPage.tsx +9 -0
  50. package/src/pages/SettingsPage.tsx +118 -0
  51. package/src/pages/WorkspacesPage.tsx +9 -0
  52. package/src/pages/index.ts +6 -0
  53. package/src/test/setup.ts +31 -0
  54. package/src/test/utils.tsx +15 -0
  55. package/src/types.ts +132 -0
  56. package/tailwind.config.js +11 -0
  57. package/tests/e2e/auth.spec.ts +42 -0
  58. package/tests/e2e/dashboard.spec.ts +52 -0
  59. package/tests/e2e/integrations.spec.ts +91 -0
  60. package/tests/e2e/workspaces.spec.ts +78 -0
  61. package/tsconfig.json +26 -0
  62. package/tsconfig.node.json +10 -0
  63. package/vite.config.ts +26 -0
  64. package/vitest.config.ts +31 -0
@@ -0,0 +1,475 @@
1
+
2
+ <!doctype html>
3
+ <html lang="en">
4
+
5
+ <head>
6
+ <title>Code coverage report for src/hooks/useWorkspace.ts</title>
7
+ <meta charset="utf-8" />
8
+ <link rel="stylesheet" href="../../prettify.css" />
9
+ <link rel="stylesheet" href="../../base.css" />
10
+ <link rel="shortcut icon" type="image/x-icon" href="../../favicon.png" />
11
+ <meta name="viewport" content="width=device-width, initial-scale=1" />
12
+ <style type='text/css'>
13
+ .coverage-summary .sorter {
14
+ background-image: url(../../sort-arrow-sprite.png);
15
+ }
16
+ </style>
17
+ </head>
18
+
19
+ <body>
20
+ <div class='wrapper'>
21
+ <div class='pad1'>
22
+ <h1><a href="../../index.html">All files</a> / <a href="index.html">src/hooks</a> useWorkspace.ts</h1>
23
+ <div class='clearfix'>
24
+
25
+ <div class='fl pad1y space-right2'>
26
+ <span class="strong">54.23% </span>
27
+ <span class="quiet">Statements</span>
28
+ <span class='fraction'>32/59</span>
29
+ </div>
30
+
31
+
32
+ <div class='fl pad1y space-right2'>
33
+ <span class="strong">25% </span>
34
+ <span class="quiet">Branches</span>
35
+ <span class='fraction'>3/12</span>
36
+ </div>
37
+
38
+
39
+ <div class='fl pad1y space-right2'>
40
+ <span class="strong">50% </span>
41
+ <span class="quiet">Functions</span>
42
+ <span class='fraction'>5/10</span>
43
+ </div>
44
+
45
+
46
+ <div class='fl pad1y space-right2'>
47
+ <span class="strong">58.18% </span>
48
+ <span class="quiet">Lines</span>
49
+ <span class='fraction'>32/55</span>
50
+ </div>
51
+
52
+
53
+ </div>
54
+ <p class="quiet">
55
+ Press <em>n</em> or <em>j</em> to go to the next uncovered block, <em>b</em>, <em>p</em> or <em>k</em> for the previous block.
56
+ </p>
57
+ <template id="filterTemplate">
58
+ <div class="quiet">
59
+ Filter:
60
+ <input type="search" id="fileSearch">
61
+ </div>
62
+ </template>
63
+ </div>
64
+ <div class='status-line medium'></div>
65
+ <pre><table class="coverage">
66
+ <tr><td class="line-count quiet"><a name='L1'></a><a href='#L1'>1</a>
67
+ <a name='L2'></a><a href='#L2'>2</a>
68
+ <a name='L3'></a><a href='#L3'>3</a>
69
+ <a name='L4'></a><a href='#L4'>4</a>
70
+ <a name='L5'></a><a href='#L5'>5</a>
71
+ <a name='L6'></a><a href='#L6'>6</a>
72
+ <a name='L7'></a><a href='#L7'>7</a>
73
+ <a name='L8'></a><a href='#L8'>8</a>
74
+ <a name='L9'></a><a href='#L9'>9</a>
75
+ <a name='L10'></a><a href='#L10'>10</a>
76
+ <a name='L11'></a><a href='#L11'>11</a>
77
+ <a name='L12'></a><a href='#L12'>12</a>
78
+ <a name='L13'></a><a href='#L13'>13</a>
79
+ <a name='L14'></a><a href='#L14'>14</a>
80
+ <a name='L15'></a><a href='#L15'>15</a>
81
+ <a name='L16'></a><a href='#L16'>16</a>
82
+ <a name='L17'></a><a href='#L17'>17</a>
83
+ <a name='L18'></a><a href='#L18'>18</a>
84
+ <a name='L19'></a><a href='#L19'>19</a>
85
+ <a name='L20'></a><a href='#L20'>20</a>
86
+ <a name='L21'></a><a href='#L21'>21</a>
87
+ <a name='L22'></a><a href='#L22'>22</a>
88
+ <a name='L23'></a><a href='#L23'>23</a>
89
+ <a name='L24'></a><a href='#L24'>24</a>
90
+ <a name='L25'></a><a href='#L25'>25</a>
91
+ <a name='L26'></a><a href='#L26'>26</a>
92
+ <a name='L27'></a><a href='#L27'>27</a>
93
+ <a name='L28'></a><a href='#L28'>28</a>
94
+ <a name='L29'></a><a href='#L29'>29</a>
95
+ <a name='L30'></a><a href='#L30'>30</a>
96
+ <a name='L31'></a><a href='#L31'>31</a>
97
+ <a name='L32'></a><a href='#L32'>32</a>
98
+ <a name='L33'></a><a href='#L33'>33</a>
99
+ <a name='L34'></a><a href='#L34'>34</a>
100
+ <a name='L35'></a><a href='#L35'>35</a>
101
+ <a name='L36'></a><a href='#L36'>36</a>
102
+ <a name='L37'></a><a href='#L37'>37</a>
103
+ <a name='L38'></a><a href='#L38'>38</a>
104
+ <a name='L39'></a><a href='#L39'>39</a>
105
+ <a name='L40'></a><a href='#L40'>40</a>
106
+ <a name='L41'></a><a href='#L41'>41</a>
107
+ <a name='L42'></a><a href='#L42'>42</a>
108
+ <a name='L43'></a><a href='#L43'>43</a>
109
+ <a name='L44'></a><a href='#L44'>44</a>
110
+ <a name='L45'></a><a href='#L45'>45</a>
111
+ <a name='L46'></a><a href='#L46'>46</a>
112
+ <a name='L47'></a><a href='#L47'>47</a>
113
+ <a name='L48'></a><a href='#L48'>48</a>
114
+ <a name='L49'></a><a href='#L49'>49</a>
115
+ <a name='L50'></a><a href='#L50'>50</a>
116
+ <a name='L51'></a><a href='#L51'>51</a>
117
+ <a name='L52'></a><a href='#L52'>52</a>
118
+ <a name='L53'></a><a href='#L53'>53</a>
119
+ <a name='L54'></a><a href='#L54'>54</a>
120
+ <a name='L55'></a><a href='#L55'>55</a>
121
+ <a name='L56'></a><a href='#L56'>56</a>
122
+ <a name='L57'></a><a href='#L57'>57</a>
123
+ <a name='L58'></a><a href='#L58'>58</a>
124
+ <a name='L59'></a><a href='#L59'>59</a>
125
+ <a name='L60'></a><a href='#L60'>60</a>
126
+ <a name='L61'></a><a href='#L61'>61</a>
127
+ <a name='L62'></a><a href='#L62'>62</a>
128
+ <a name='L63'></a><a href='#L63'>63</a>
129
+ <a name='L64'></a><a href='#L64'>64</a>
130
+ <a name='L65'></a><a href='#L65'>65</a>
131
+ <a name='L66'></a><a href='#L66'>66</a>
132
+ <a name='L67'></a><a href='#L67'>67</a>
133
+ <a name='L68'></a><a href='#L68'>68</a>
134
+ <a name='L69'></a><a href='#L69'>69</a>
135
+ <a name='L70'></a><a href='#L70'>70</a>
136
+ <a name='L71'></a><a href='#L71'>71</a>
137
+ <a name='L72'></a><a href='#L72'>72</a>
138
+ <a name='L73'></a><a href='#L73'>73</a>
139
+ <a name='L74'></a><a href='#L74'>74</a>
140
+ <a name='L75'></a><a href='#L75'>75</a>
141
+ <a name='L76'></a><a href='#L76'>76</a>
142
+ <a name='L77'></a><a href='#L77'>77</a>
143
+ <a name='L78'></a><a href='#L78'>78</a>
144
+ <a name='L79'></a><a href='#L79'>79</a>
145
+ <a name='L80'></a><a href='#L80'>80</a>
146
+ <a name='L81'></a><a href='#L81'>81</a>
147
+ <a name='L82'></a><a href='#L82'>82</a>
148
+ <a name='L83'></a><a href='#L83'>83</a>
149
+ <a name='L84'></a><a href='#L84'>84</a>
150
+ <a name='L85'></a><a href='#L85'>85</a>
151
+ <a name='L86'></a><a href='#L86'>86</a>
152
+ <a name='L87'></a><a href='#L87'>87</a>
153
+ <a name='L88'></a><a href='#L88'>88</a>
154
+ <a name='L89'></a><a href='#L89'>89</a>
155
+ <a name='L90'></a><a href='#L90'>90</a>
156
+ <a name='L91'></a><a href='#L91'>91</a>
157
+ <a name='L92'></a><a href='#L92'>92</a>
158
+ <a name='L93'></a><a href='#L93'>93</a>
159
+ <a name='L94'></a><a href='#L94'>94</a>
160
+ <a name='L95'></a><a href='#L95'>95</a>
161
+ <a name='L96'></a><a href='#L96'>96</a>
162
+ <a name='L97'></a><a href='#L97'>97</a>
163
+ <a name='L98'></a><a href='#L98'>98</a>
164
+ <a name='L99'></a><a href='#L99'>99</a>
165
+ <a name='L100'></a><a href='#L100'>100</a>
166
+ <a name='L101'></a><a href='#L101'>101</a>
167
+ <a name='L102'></a><a href='#L102'>102</a>
168
+ <a name='L103'></a><a href='#L103'>103</a>
169
+ <a name='L104'></a><a href='#L104'>104</a>
170
+ <a name='L105'></a><a href='#L105'>105</a>
171
+ <a name='L106'></a><a href='#L106'>106</a>
172
+ <a name='L107'></a><a href='#L107'>107</a>
173
+ <a name='L108'></a><a href='#L108'>108</a>
174
+ <a name='L109'></a><a href='#L109'>109</a>
175
+ <a name='L110'></a><a href='#L110'>110</a>
176
+ <a name='L111'></a><a href='#L111'>111</a>
177
+ <a name='L112'></a><a href='#L112'>112</a>
178
+ <a name='L113'></a><a href='#L113'>113</a>
179
+ <a name='L114'></a><a href='#L114'>114</a>
180
+ <a name='L115'></a><a href='#L115'>115</a>
181
+ <a name='L116'></a><a href='#L116'>116</a>
182
+ <a name='L117'></a><a href='#L117'>117</a>
183
+ <a name='L118'></a><a href='#L118'>118</a>
184
+ <a name='L119'></a><a href='#L119'>119</a>
185
+ <a name='L120'></a><a href='#L120'>120</a>
186
+ <a name='L121'></a><a href='#L121'>121</a>
187
+ <a name='L122'></a><a href='#L122'>122</a>
188
+ <a name='L123'></a><a href='#L123'>123</a>
189
+ <a name='L124'></a><a href='#L124'>124</a>
190
+ <a name='L125'></a><a href='#L125'>125</a>
191
+ <a name='L126'></a><a href='#L126'>126</a>
192
+ <a name='L127'></a><a href='#L127'>127</a>
193
+ <a name='L128'></a><a href='#L128'>128</a>
194
+ <a name='L129'></a><a href='#L129'>129</a>
195
+ <a name='L130'></a><a href='#L130'>130</a>
196
+ <a name='L131'></a><a href='#L131'>131</a></td><td class="line-coverage quiet"><span class="cline-any cline-neutral">&nbsp;</span>
197
+ <span class="cline-any cline-neutral">&nbsp;</span>
198
+ <span class="cline-any cline-neutral">&nbsp;</span>
199
+ <span class="cline-any cline-neutral">&nbsp;</span>
200
+ <span class="cline-any cline-neutral">&nbsp;</span>
201
+ <span class="cline-any cline-neutral">&nbsp;</span>
202
+ <span class="cline-any cline-neutral">&nbsp;</span>
203
+ <span class="cline-any cline-neutral">&nbsp;</span>
204
+ <span class="cline-any cline-neutral">&nbsp;</span>
205
+ <span class="cline-any cline-neutral">&nbsp;</span>
206
+ <span class="cline-any cline-neutral">&nbsp;</span>
207
+ <span class="cline-any cline-neutral">&nbsp;</span>
208
+ <span class="cline-any cline-neutral">&nbsp;</span>
209
+ <span class="cline-any cline-neutral">&nbsp;</span>
210
+ <span class="cline-any cline-neutral">&nbsp;</span>
211
+ <span class="cline-any cline-neutral">&nbsp;</span>
212
+ <span class="cline-any cline-neutral">&nbsp;</span>
213
+ <span class="cline-any cline-neutral">&nbsp;</span>
214
+ <span class="cline-any cline-yes">1x</span>
215
+ <span class="cline-any cline-yes">15x</span>
216
+ <span class="cline-any cline-yes">15x</span>
217
+ <span class="cline-any cline-yes">15x</span>
218
+ <span class="cline-any cline-yes">15x</span>
219
+ <span class="cline-any cline-yes">15x</span>
220
+ <span class="cline-any cline-yes">15x</span>
221
+ <span class="cline-any cline-neutral">&nbsp;</span>
222
+ <span class="cline-any cline-neutral">&nbsp;</span>
223
+ <span class="cline-any cline-yes">15x</span>
224
+ <span class="cline-any cline-yes">4x</span>
225
+ <span class="cline-any cline-yes">4x</span>
226
+ <span class="cline-any cline-yes">4x</span>
227
+ <span class="cline-any cline-yes">4x</span>
228
+ <span class="cline-any cline-neutral">&nbsp;</span>
229
+ <span class="cline-any cline-neutral">&nbsp;</span>
230
+ <span class="cline-any cline-neutral">&nbsp;</span>
231
+ <span class="cline-any cline-yes">4x</span>
232
+ <span class="cline-any cline-yes">4x</span>
233
+ <span class="cline-any cline-neutral">&nbsp;</span>
234
+ <span class="cline-any cline-neutral">&nbsp;</span>
235
+ <span class="cline-any cline-yes">4x</span>
236
+ <span class="cline-any cline-yes">4x</span>
237
+ <span class="cline-any cline-neutral">&nbsp;</span>
238
+ <span class="cline-any cline-neutral">&nbsp;</span>
239
+ <span class="cline-any cline-no">&nbsp;</span>
240
+ <span class="cline-any cline-neutral">&nbsp;</span>
241
+ <span class="cline-any cline-yes">4x</span>
242
+ <span class="cline-any cline-neutral">&nbsp;</span>
243
+ <span class="cline-any cline-neutral">&nbsp;</span>
244
+ <span class="cline-any cline-neutral">&nbsp;</span>
245
+ <span class="cline-any cline-yes">4x</span>
246
+ <span class="cline-any cline-neutral">&nbsp;</span>
247
+ <span class="cline-any cline-neutral">&nbsp;</span>
248
+ <span class="cline-any cline-neutral">&nbsp;</span>
249
+ <span class="cline-any cline-yes">15x</span>
250
+ <span class="cline-any cline-yes">7x</span>
251
+ <span class="cline-any cline-yes">7x</span>
252
+ <span class="cline-any cline-yes">4x</span>
253
+ <span class="cline-any cline-yes">4x</span>
254
+ <span class="cline-any cline-neutral">&nbsp;</span>
255
+ <span class="cline-any cline-neutral">&nbsp;</span>
256
+ <span class="cline-any cline-yes">3x</span>
257
+ <span class="cline-any cline-yes">3x</span>
258
+ <span class="cline-any cline-yes">3x</span>
259
+ <span class="cline-any cline-neutral">&nbsp;</span>
260
+ <span class="cline-any cline-no">&nbsp;</span>
261
+ <span class="cline-any cline-neutral">&nbsp;</span>
262
+ <span class="cline-any cline-neutral">&nbsp;</span>
263
+ <span class="cline-any cline-neutral">&nbsp;</span>
264
+ <span class="cline-any cline-yes">7x</span>
265
+ <span class="cline-any cline-neutral">&nbsp;</span>
266
+ <span class="cline-any cline-neutral">&nbsp;</span>
267
+ <span class="cline-any cline-yes">15x</span>
268
+ <span class="cline-any cline-no">&nbsp;</span>
269
+ <span class="cline-any cline-no">&nbsp;</span>
270
+ <span class="cline-any cline-no">&nbsp;</span>
271
+ <span class="cline-any cline-neutral">&nbsp;</span>
272
+ <span class="cline-any cline-no">&nbsp;</span>
273
+ <span class="cline-any cline-neutral">&nbsp;</span>
274
+ <span class="cline-any cline-neutral">&nbsp;</span>
275
+ <span class="cline-any cline-neutral">&nbsp;</span>
276
+ <span class="cline-any cline-yes">15x</span>
277
+ <span class="cline-any cline-no">&nbsp;</span>
278
+ <span class="cline-any cline-neutral">&nbsp;</span>
279
+ <span class="cline-any cline-no">&nbsp;</span>
280
+ <span class="cline-any cline-no">&nbsp;</span>
281
+ <span class="cline-any cline-neutral">&nbsp;</span>
282
+ <span class="cline-any cline-no">&nbsp;</span>
283
+ <span class="cline-any cline-no">&nbsp;</span>
284
+ <span class="cline-any cline-neutral">&nbsp;</span>
285
+ <span class="cline-any cline-no">&nbsp;</span>
286
+ <span class="cline-any cline-neutral">&nbsp;</span>
287
+ <span class="cline-any cline-neutral">&nbsp;</span>
288
+ <span class="cline-any cline-neutral">&nbsp;</span>
289
+ <span class="cline-any cline-yes">15x</span>
290
+ <span class="cline-any cline-no">&nbsp;</span>
291
+ <span class="cline-any cline-neutral">&nbsp;</span>
292
+ <span class="cline-any cline-no">&nbsp;</span>
293
+ <span class="cline-any cline-no">&nbsp;</span>
294
+ <span class="cline-any cline-no">&nbsp;</span>
295
+ <span class="cline-any cline-neutral">&nbsp;</span>
296
+ <span class="cline-any cline-no">&nbsp;</span>
297
+ <span class="cline-any cline-neutral">&nbsp;</span>
298
+ <span class="cline-any cline-neutral">&nbsp;</span>
299
+ <span class="cline-any cline-neutral">&nbsp;</span>
300
+ <span class="cline-any cline-yes">15x</span>
301
+ <span class="cline-any cline-no">&nbsp;</span>
302
+ <span class="cline-any cline-neutral">&nbsp;</span>
303
+ <span class="cline-any cline-no">&nbsp;</span>
304
+ <span class="cline-any cline-no">&nbsp;</span>
305
+ <span class="cline-any cline-no">&nbsp;</span>
306
+ <span class="cline-any cline-no">&nbsp;</span>
307
+ <span class="cline-any cline-neutral">&nbsp;</span>
308
+ <span class="cline-any cline-no">&nbsp;</span>
309
+ <span class="cline-any cline-neutral">&nbsp;</span>
310
+ <span class="cline-any cline-neutral">&nbsp;</span>
311
+ <span class="cline-any cline-neutral">&nbsp;</span>
312
+ <span class="cline-any cline-yes">15x</span>
313
+ <span class="cline-any cline-neutral">&nbsp;</span>
314
+ <span class="cline-any cline-neutral">&nbsp;</span>
315
+ <span class="cline-any cline-neutral">&nbsp;</span>
316
+ <span class="cline-any cline-neutral">&nbsp;</span>
317
+ <span class="cline-any cline-neutral">&nbsp;</span>
318
+ <span class="cline-any cline-neutral">&nbsp;</span>
319
+ <span class="cline-any cline-neutral">&nbsp;</span>
320
+ <span class="cline-any cline-neutral">&nbsp;</span>
321
+ <span class="cline-any cline-neutral">&nbsp;</span>
322
+ <span class="cline-any cline-neutral">&nbsp;</span>
323
+ <span class="cline-any cline-neutral">&nbsp;</span>
324
+ <span class="cline-any cline-neutral">&nbsp;</span>
325
+ <span class="cline-any cline-neutral">&nbsp;</span>
326
+ <span class="cline-any cline-neutral">&nbsp;</span></td><td class="text"><pre class="prettyprint lang-js">import { useState, useEffect, useCallback } from 'react'
327
+ import { workspaceAPI, rbacAPI } from '../api'
328
+ import * as types from '../types'
329
+ &nbsp;
330
+ interface UseWorkspaceReturn {
331
+ workspaces: types.Workspace[]
332
+ currentWorkspace: types.Workspace | null
333
+ members: types.WorkspaceMember[]
334
+ roles: types.Role[]
335
+ isLoading: boolean
336
+ error: string | null
337
+ setCurrentWorkspace: (workspace: types.Workspace | null) =&gt; void
338
+ createWorkspace: (name: string, isPublic: boolean) =&gt; Promise&lt;void&gt;
339
+ addMember: (userId: string, roleName: string) =&gt; Promise&lt;void&gt;
340
+ removeMember: (userId: string) =&gt; Promise&lt;void&gt;
341
+ updateMember: (userId: string, roleName: string) =&gt; Promise&lt;void&gt;
342
+ }
343
+ &nbsp;
344
+ export const useWorkspace = (): UseWorkspaceReturn =&gt; {
345
+ const [workspaces, setWorkspaces] = useState&lt;types.Workspace[]&gt;([])
346
+ const [currentWorkspace, setCurrentWorkspace] = useState&lt;types.Workspace | null&gt;(null)
347
+ const [members, setMembers] = useState&lt;types.WorkspaceMember[]&gt;([])
348
+ const [roles, setRoles] = useState&lt;types.Role[]&gt;([])
349
+ const [isLoading, setIsLoading] = useState(true)
350
+ const [error, setError] = useState&lt;string | null&gt;(null)
351
+ &nbsp;
352
+ // Fetch workspaces and roles
353
+ useEffect(() =&gt; {
354
+ const loadData = async () =&gt; {
355
+ try {
356
+ setIsLoading(true)
357
+ const [workspacesRes, rolesRes] = await Promise.all([
358
+ workspaceAPI.list(),
359
+ rbacAPI.getRoles()
360
+ ])
361
+ setWorkspaces(workspacesRes.data)
362
+ setRoles(rolesRes.data)
363
+ &nbsp;
364
+ // Set first workspace as current
365
+ <span class="missing-if-branch" title="else path not taken" >E</span>if (workspacesRes.data.length &gt; 0) {
366
+ setCurrentWorkspace(workspacesRes.data[0])
367
+ }
368
+ } catch (err) {
369
+ <span class="cstat-no" title="statement not covered" > setError(err instanceof Error ? err.message : 'Failed to load workspaces')</span>
370
+ } finally {
371
+ setIsLoading(false)
372
+ }
373
+ }
374
+ &nbsp;
375
+ loadData()
376
+ }, [])
377
+ &nbsp;
378
+ // Fetch members when workspace changes
379
+ useEffect(() =&gt; {
380
+ const loadMembers = async () =&gt; {
381
+ if (!currentWorkspace) {
382
+ setMembers([])
383
+ return
384
+ }
385
+ &nbsp;
386
+ try {
387
+ const res = await workspaceAPI.getMembers(currentWorkspace.id)
388
+ setMembers(res.data)
389
+ } catch (err) {
390
+ <span class="cstat-no" title="statement not covered" > console.error('Failed to load members:', err)</span>
391
+ }
392
+ }
393
+ &nbsp;
394
+ loadMembers()
395
+ }, [currentWorkspace])
396
+ &nbsp;
397
+ const createWorkspace = useCallback(<span class="fstat-no" title="function not covered" >async (n</span>ame: string, isPublic: boolean) =&gt; {
398
+ <span class="cstat-no" title="statement not covered" > try {</span>
399
+ const res = <span class="cstat-no" title="statement not covered" >await workspaceAPI.create(name, isPublic)</span>
400
+ <span class="cstat-no" title="statement not covered" > setWorkspaces([...workspaces, res.data])</span>
401
+ } catch (err) {
402
+ <span class="cstat-no" title="statement not covered" > throw err</span>
403
+ }
404
+ }, [workspaces])
405
+ &nbsp;
406
+ const addMember = useCallback(<span class="fstat-no" title="function not covered" >async (u</span>serId: string, roleName: string) =&gt; {
407
+ <span class="cstat-no" title="statement not covered" > if (!currentWorkspace) <span class="cstat-no" title="statement not covered" >throw new Error('No workspace selected')</span></span>
408
+ &nbsp;
409
+ <span class="cstat-no" title="statement not covered" > try {</span>
410
+ <span class="cstat-no" title="statement not covered" > await workspaceAPI.addMember(currentWorkspace.id, userId, roleName)</span>
411
+ // Refresh members
412
+ const res = <span class="cstat-no" title="statement not covered" >await workspaceAPI.getMembers(currentWorkspace.id)</span>
413
+ <span class="cstat-no" title="statement not covered" > setMembers(res.data)</span>
414
+ } catch (err) {
415
+ <span class="cstat-no" title="statement not covered" > throw err</span>
416
+ }
417
+ }, [currentWorkspace])
418
+ &nbsp;
419
+ const removeMember = useCallback(<span class="fstat-no" title="function not covered" >async (u</span>serId: string) =&gt; {
420
+ <span class="cstat-no" title="statement not covered" > if (!currentWorkspace) <span class="cstat-no" title="statement not covered" >throw new Error('No workspace selected')</span></span>
421
+ &nbsp;
422
+ <span class="cstat-no" title="statement not covered" > try {</span>
423
+ <span class="cstat-no" title="statement not covered" > await workspaceAPI.removeMember(currentWorkspace.id, userId)</span>
424
+ <span class="cstat-no" title="statement not covered" > setMembers(members.<span class="fstat-no" title="function not covered" >filter(m</span> =&gt; <span class="cstat-no" title="statement not covered" >m.id !== userId))</span></span>
425
+ } catch (err) {
426
+ <span class="cstat-no" title="statement not covered" > throw err</span>
427
+ }
428
+ }, [currentWorkspace, members])
429
+ &nbsp;
430
+ const updateMember = useCallback(<span class="fstat-no" title="function not covered" >async (u</span>serId: string, roleName: string) =&gt; {
431
+ <span class="cstat-no" title="statement not covered" > if (!currentWorkspace) <span class="cstat-no" title="statement not covered" >throw new Error('No workspace selected')</span></span>
432
+ &nbsp;
433
+ <span class="cstat-no" title="statement not covered" > try {</span>
434
+ <span class="cstat-no" title="statement not covered" > await workspaceAPI.updateMember(currentWorkspace.id, userId, roleName)</span>
435
+ const res = <span class="cstat-no" title="statement not covered" >await workspaceAPI.getMembers(currentWorkspace.id)</span>
436
+ <span class="cstat-no" title="statement not covered" > setMembers(res.data)</span>
437
+ } catch (err) {
438
+ <span class="cstat-no" title="statement not covered" > throw err</span>
439
+ }
440
+ }, [currentWorkspace])
441
+ &nbsp;
442
+ return {
443
+ workspaces,
444
+ currentWorkspace,
445
+ members,
446
+ roles,
447
+ isLoading,
448
+ error,
449
+ setCurrentWorkspace,
450
+ createWorkspace,
451
+ addMember,
452
+ removeMember,
453
+ updateMember
454
+ }
455
+ }
456
+ &nbsp;</pre></td></tr></table></pre>
457
+
458
+ <div class='push'></div><!-- for sticky footer -->
459
+ </div><!-- /wrapper -->
460
+ <div class='footer quiet pad2 space-top1 center small'>
461
+ Code coverage generated by
462
+ <a href="https://istanbul.js.org/" target="_blank" rel="noopener noreferrer">istanbul</a>
463
+ at 2026-03-27T04:52:03.111Z
464
+ </div>
465
+ <script src="../../prettify.js"></script>
466
+ <script>
467
+ window.onload = function () {
468
+ prettyPrint();
469
+ };
470
+ </script>
471
+ <script src="../../sorter.js"></script>
472
+ <script src="../../block-navigation.js"></script>
473
+ </body>
474
+ </html>
475
+
@@ -0,0 +1,116 @@
1
+
2
+ <!doctype html>
3
+ <html lang="en">
4
+
5
+ <head>
6
+ <title>Code coverage report for src</title>
7
+ <meta charset="utf-8" />
8
+ <link rel="stylesheet" href="../prettify.css" />
9
+ <link rel="stylesheet" href="../base.css" />
10
+ <link rel="shortcut icon" type="image/x-icon" href="../favicon.png" />
11
+ <meta name="viewport" content="width=device-width, initial-scale=1" />
12
+ <style type='text/css'>
13
+ .coverage-summary .sorter {
14
+ background-image: url(../sort-arrow-sprite.png);
15
+ }
16
+ </style>
17
+ </head>
18
+
19
+ <body>
20
+ <div class='wrapper'>
21
+ <div class='pad1'>
22
+ <h1><a href="../index.html">All files</a> src</h1>
23
+ <div class='clearfix'>
24
+
25
+ <div class='fl pad1y space-right2'>
26
+ <span class="strong">26.92% </span>
27
+ <span class="quiet">Statements</span>
28
+ <span class='fraction'>7/26</span>
29
+ </div>
30
+
31
+
32
+ <div class='fl pad1y space-right2'>
33
+ <span class="strong">33.33% </span>
34
+ <span class="quiet">Branches</span>
35
+ <span class='fraction'>2/6</span>
36
+ </div>
37
+
38
+
39
+ <div class='fl pad1y space-right2'>
40
+ <span class="strong">0% </span>
41
+ <span class="quiet">Functions</span>
42
+ <span class='fraction'>0/6</span>
43
+ </div>
44
+
45
+
46
+ <div class='fl pad1y space-right2'>
47
+ <span class="strong">26.92% </span>
48
+ <span class="quiet">Lines</span>
49
+ <span class='fraction'>7/26</span>
50
+ </div>
51
+
52
+
53
+ </div>
54
+ <p class="quiet">
55
+ Press <em>n</em> or <em>j</em> to go to the next uncovered block, <em>b</em>, <em>p</em> or <em>k</em> for the previous block.
56
+ </p>
57
+ <template id="filterTemplate">
58
+ <div class="quiet">
59
+ Filter:
60
+ <input type="search" id="fileSearch">
61
+ </div>
62
+ </template>
63
+ </div>
64
+ <div class='status-line low'></div>
65
+ <div class="pad1">
66
+ <table class="coverage-summary">
67
+ <thead>
68
+ <tr>
69
+ <th data-col="file" data-fmt="html" data-html="true" class="file">File</th>
70
+ <th data-col="pic" data-type="number" data-fmt="html" data-html="true" class="pic"></th>
71
+ <th data-col="statements" data-type="number" data-fmt="pct" class="pct">Statements</th>
72
+ <th data-col="statements_raw" data-type="number" data-fmt="html" class="abs"></th>
73
+ <th data-col="branches" data-type="number" data-fmt="pct" class="pct">Branches</th>
74
+ <th data-col="branches_raw" data-type="number" data-fmt="html" class="abs"></th>
75
+ <th data-col="functions" data-type="number" data-fmt="pct" class="pct">Functions</th>
76
+ <th data-col="functions_raw" data-type="number" data-fmt="html" class="abs"></th>
77
+ <th data-col="lines" data-type="number" data-fmt="pct" class="pct">Lines</th>
78
+ <th data-col="lines_raw" data-type="number" data-fmt="html" class="abs"></th>
79
+ </tr>
80
+ </thead>
81
+ <tbody><tr>
82
+ <td class="file low" data-value="auth.ts"><a href="auth.ts.html">auth.ts</a></td>
83
+ <td data-value="26.92" class="pic low">
84
+ <div class="chart"><div class="cover-fill" style="width: 26%"></div><div class="cover-empty" style="width: 74%"></div></div>
85
+ </td>
86
+ <td data-value="26.92" class="pct low">26.92%</td>
87
+ <td data-value="26" class="abs low">7/26</td>
88
+ <td data-value="33.33" class="pct low">33.33%</td>
89
+ <td data-value="6" class="abs low">2/6</td>
90
+ <td data-value="0" class="pct low">0%</td>
91
+ <td data-value="6" class="abs low">0/6</td>
92
+ <td data-value="26.92" class="pct low">26.92%</td>
93
+ <td data-value="26" class="abs low">7/26</td>
94
+ </tr>
95
+
96
+ </tbody>
97
+ </table>
98
+ </div>
99
+ <div class='push'></div><!-- for sticky footer -->
100
+ </div><!-- /wrapper -->
101
+ <div class='footer quiet pad2 space-top1 center small'>
102
+ Code coverage generated by
103
+ <a href="https://istanbul.js.org/" target="_blank" rel="noopener noreferrer">istanbul</a>
104
+ at 2026-03-27T04:52:03.111Z
105
+ </div>
106
+ <script src="../prettify.js"></script>
107
+ <script>
108
+ window.onload = function () {
109
+ prettyPrint();
110
+ };
111
+ </script>
112
+ <script src="../sorter.js"></script>
113
+ <script src="../block-navigation.js"></script>
114
+ </body>
115
+ </html>
116
+
package/index.html ADDED
@@ -0,0 +1,13 @@
1
+ <!doctype html>
2
+ <html lang="zh-Hant">
3
+ <head>
4
+ <meta charset="UTF-8" />
5
+ <link rel="icon" type="image/svg+xml" href="/vite.svg" />
6
+ <meta name="viewport" content="width=device-width, initial-scale=1.0" />
7
+ <title>Session Wrap - Dashboard</title>
8
+ </head>
9
+ <body>
10
+ <div id="root"></div>
11
+ <script type="module" src="/src/main.tsx"></script>
12
+ </body>
13
+ </html>