@frictionless-ts/dataset 1.0.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 (242) hide show
  1. package/README.md +3 -0
  2. package/build/file/copy.d.ts +4 -0
  3. package/build/file/copy.js +7 -0
  4. package/build/file/copy.spec.d.ts +1 -0
  5. package/build/file/copy.spec.js +104 -0
  6. package/build/file/describe.d.ts +7 -0
  7. package/build/file/describe.js +9 -0
  8. package/build/file/fetch.d.ts +2 -0
  9. package/build/file/fetch.js +21 -0
  10. package/build/file/index.d.ts +9 -0
  11. package/build/file/index.js +10 -0
  12. package/build/file/infer.d.ts +10 -0
  13. package/build/file/infer.js +48 -0
  14. package/build/file/infer.spec.d.ts +1 -0
  15. package/build/file/infer.spec.js +141 -0
  16. package/build/file/load.d.ts +3 -0
  17. package/build/file/load.js +7 -0
  18. package/build/file/path.d.ts +2 -0
  19. package/build/file/path.js +17 -0
  20. package/build/file/save.d.ts +3 -0
  21. package/build/file/save.js +7 -0
  22. package/build/file/temp.d.ts +11 -0
  23. package/build/file/temp.js +23 -0
  24. package/build/file/validate.d.ts +6 -0
  25. package/build/file/validate.js +47 -0
  26. package/build/file/validate.spec.d.ts +1 -0
  27. package/build/file/validate.spec.js +180 -0
  28. package/build/folder/create.d.ts +1 -0
  29. package/build/folder/create.js +5 -0
  30. package/build/folder/index.d.ts +2 -0
  31. package/build/folder/index.js +3 -0
  32. package/build/folder/temp.d.ts +3 -0
  33. package/build/folder/temp.js +16 -0
  34. package/build/index.d.ts +36 -0
  35. package/build/index.js +35 -0
  36. package/build/package/index.d.ts +2 -0
  37. package/build/package/index.js +3 -0
  38. package/build/package/merge.d.ts +23 -0
  39. package/build/package/merge.js +12 -0
  40. package/build/package/path.d.ts +3 -0
  41. package/build/package/path.js +43 -0
  42. package/build/package/path.spec.d.ts +1 -0
  43. package/build/package/path.spec.js +56 -0
  44. package/build/plugin.d.ts +11 -0
  45. package/build/plugin.js +2 -0
  46. package/build/plugins/ckan/ckan/index.d.ts +1 -0
  47. package/build/plugins/ckan/ckan/index.js +2 -0
  48. package/build/plugins/ckan/ckan/request.d.ts +11 -0
  49. package/build/plugins/ckan/ckan/request.js +35 -0
  50. package/build/plugins/ckan/index.d.ts +4 -0
  51. package/build/plugins/ckan/index.js +5 -0
  52. package/build/plugins/ckan/package/Organization.d.ts +21 -0
  53. package/build/plugins/ckan/package/Organization.js +2 -0
  54. package/build/plugins/ckan/package/Package.d.ts +76 -0
  55. package/build/plugins/ckan/package/Package.js +2 -0
  56. package/build/plugins/ckan/package/Tag.d.ts +17 -0
  57. package/build/plugins/ckan/package/Tag.js +2 -0
  58. package/build/plugins/ckan/package/convert/fromCkan.d.ts +3 -0
  59. package/build/plugins/ckan/package/convert/fromCkan.js +63 -0
  60. package/build/plugins/ckan/package/convert/fromCkan.spec.d.ts +1 -0
  61. package/build/plugins/ckan/package/convert/fromCkan.spec.js +80 -0
  62. package/build/plugins/ckan/package/convert/toCkan.d.ts +22 -0
  63. package/build/plugins/ckan/package/convert/toCkan.js +51 -0
  64. package/build/plugins/ckan/package/convert/toCkan.spec.d.ts +1 -0
  65. package/build/plugins/ckan/package/convert/toCkan.spec.js +153 -0
  66. package/build/plugins/ckan/package/fixtures/ckan-package.json +308 -0
  67. package/build/plugins/ckan/package/index.d.ts +7 -0
  68. package/build/plugins/ckan/package/index.js +5 -0
  69. package/build/plugins/ckan/package/load.d.ts +21 -0
  70. package/build/plugins/ckan/package/load.js +74 -0
  71. package/build/plugins/ckan/package/load.spec.d.ts +1 -0
  72. package/build/plugins/ckan/package/load.spec.js +11 -0
  73. package/build/plugins/ckan/package/save.d.ts +10 -0
  74. package/build/plugins/ckan/package/save.js +83 -0
  75. package/build/plugins/ckan/package/save.spec.d.ts +1 -0
  76. package/build/plugins/ckan/package/save.spec.js +319 -0
  77. package/build/plugins/ckan/plugin.d.ts +19 -0
  78. package/build/plugins/ckan/plugin.js +18 -0
  79. package/build/plugins/ckan/plugin.spec.d.ts +1 -0
  80. package/build/plugins/ckan/plugin.spec.js +83 -0
  81. package/build/plugins/ckan/resource/Resource.d.ts +56 -0
  82. package/build/plugins/ckan/resource/Resource.js +2 -0
  83. package/build/plugins/ckan/resource/convert/fromCkan.d.ts +3 -0
  84. package/build/plugins/ckan/resource/convert/fromCkan.js +39 -0
  85. package/build/plugins/ckan/resource/convert/toCkan.d.ts +3 -0
  86. package/build/plugins/ckan/resource/convert/toCkan.js +20 -0
  87. package/build/plugins/ckan/resource/index.d.ts +3 -0
  88. package/build/plugins/ckan/resource/index.js +3 -0
  89. package/build/plugins/ckan/schema/Field.d.ts +34 -0
  90. package/build/plugins/ckan/schema/Field.js +2 -0
  91. package/build/plugins/ckan/schema/Schema.d.ts +10 -0
  92. package/build/plugins/ckan/schema/Schema.js +2 -0
  93. package/build/plugins/ckan/schema/convert/fixtures/ckan-schema.json +115 -0
  94. package/build/plugins/ckan/schema/convert/fromCkan.d.ts +3 -0
  95. package/build/plugins/ckan/schema/convert/fromCkan.js +47 -0
  96. package/build/plugins/ckan/schema/convert/fromCkan.spec.d.ts +1 -0
  97. package/build/plugins/ckan/schema/convert/fromCkan.spec.js +157 -0
  98. package/build/plugins/ckan/schema/convert/toCkan.d.ts +3 -0
  99. package/build/plugins/ckan/schema/convert/toCkan.js +50 -0
  100. package/build/plugins/ckan/schema/convert/toCkan.spec.d.ts +1 -0
  101. package/build/plugins/ckan/schema/convert/toCkan.spec.js +278 -0
  102. package/build/plugins/ckan/schema/index.d.ts +4 -0
  103. package/build/plugins/ckan/schema/index.js +3 -0
  104. package/build/plugins/datahub/index.d.ts +2 -0
  105. package/build/plugins/datahub/index.js +3 -0
  106. package/build/plugins/datahub/package/index.d.ts +1 -0
  107. package/build/plugins/datahub/package/index.js +2 -0
  108. package/build/plugins/datahub/package/load.d.ts +1 -0
  109. package/build/plugins/datahub/package/load.js +9 -0
  110. package/build/plugins/datahub/package/load.spec.d.ts +1 -0
  111. package/build/plugins/datahub/package/load.spec.js +11 -0
  112. package/build/plugins/datahub/plugin.d.ts +4 -0
  113. package/build/plugins/datahub/plugin.js +18 -0
  114. package/build/plugins/datahub/plugin.spec.d.ts +1 -0
  115. package/build/plugins/datahub/plugin.spec.js +78 -0
  116. package/build/plugins/descriptor/index.d.ts +1 -0
  117. package/build/plugins/descriptor/index.js +2 -0
  118. package/build/plugins/descriptor/plugin.d.ts +11 -0
  119. package/build/plugins/descriptor/plugin.js +29 -0
  120. package/build/plugins/descriptor/plugin.spec.d.ts +1 -0
  121. package/build/plugins/descriptor/plugin.spec.js +169 -0
  122. package/build/plugins/folder/index.d.ts +2 -0
  123. package/build/plugins/folder/index.js +3 -0
  124. package/build/plugins/folder/package/index.d.ts +2 -0
  125. package/build/plugins/folder/package/index.js +3 -0
  126. package/build/plugins/folder/package/load.d.ts +1 -0
  127. package/build/plugins/folder/package/load.js +6 -0
  128. package/build/plugins/folder/package/load.spec.d.ts +1 -0
  129. package/build/plugins/folder/package/load.spec.js +175 -0
  130. package/build/plugins/folder/package/save.d.ts +7 -0
  131. package/build/plugins/folder/package/save.js +35 -0
  132. package/build/plugins/folder/package/save.spec.d.ts +1 -0
  133. package/build/plugins/folder/package/save.spec.js +328 -0
  134. package/build/plugins/folder/plugin.d.ts +4 -0
  135. package/build/plugins/folder/plugin.js +19 -0
  136. package/build/plugins/folder/plugin.spec.d.ts +1 -0
  137. package/build/plugins/folder/plugin.spec.js +53 -0
  138. package/build/plugins/github/github/index.d.ts +1 -0
  139. package/build/plugins/github/github/index.js +2 -0
  140. package/build/plugins/github/github/path.d.ts +1 -0
  141. package/build/plugins/github/github/path.js +4 -0
  142. package/build/plugins/github/github/request.d.ts +15 -0
  143. package/build/plugins/github/github/request.js +43 -0
  144. package/build/plugins/github/index.d.ts +3 -0
  145. package/build/plugins/github/index.js +4 -0
  146. package/build/plugins/github/package/License.d.ts +21 -0
  147. package/build/plugins/github/package/License.js +2 -0
  148. package/build/plugins/github/package/Owner.d.ts +25 -0
  149. package/build/plugins/github/package/Owner.js +2 -0
  150. package/build/plugins/github/package/Package.d.ts +87 -0
  151. package/build/plugins/github/package/Package.js +2 -0
  152. package/build/plugins/github/package/convert/fromGithub.d.ts +3 -0
  153. package/build/plugins/github/package/convert/fromGithub.js +50 -0
  154. package/build/plugins/github/package/index.d.ts +6 -0
  155. package/build/plugins/github/package/index.js +4 -0
  156. package/build/plugins/github/package/load.d.ts +23 -0
  157. package/build/plugins/github/package/load.js +48 -0
  158. package/build/plugins/github/package/load.spec.d.ts +1 -0
  159. package/build/plugins/github/package/load.spec.js +15 -0
  160. package/build/plugins/github/package/save.d.ts +14 -0
  161. package/build/plugins/github/package/save.js +69 -0
  162. package/build/plugins/github/package/save.spec.d.ts +1 -0
  163. package/build/plugins/github/package/save.spec.js +566 -0
  164. package/build/plugins/github/plugin.d.ts +19 -0
  165. package/build/plugins/github/plugin.js +18 -0
  166. package/build/plugins/github/plugin.spec.d.ts +1 -0
  167. package/build/plugins/github/plugin.spec.js +73 -0
  168. package/build/plugins/github/resource/Resource.d.ts +29 -0
  169. package/build/plugins/github/resource/Resource.js +2 -0
  170. package/build/plugins/github/resource/convert/fromGithub.d.ts +5 -0
  171. package/build/plugins/github/resource/convert/fromGithub.js +24 -0
  172. package/build/plugins/github/resource/convert/toGithub.d.ts +3 -0
  173. package/build/plugins/github/resource/convert/toGithub.js +10 -0
  174. package/build/plugins/github/resource/index.d.ts +3 -0
  175. package/build/plugins/github/resource/index.js +3 -0
  176. package/build/plugins/zenodo/index.d.ts +3 -0
  177. package/build/plugins/zenodo/index.js +4 -0
  178. package/build/plugins/zenodo/package/Creator.d.ts +20 -0
  179. package/build/plugins/zenodo/package/Creator.js +2 -0
  180. package/build/plugins/zenodo/package/Package.d.ts +94 -0
  181. package/build/plugins/zenodo/package/Package.js +2 -0
  182. package/build/plugins/zenodo/package/convert/fromZenodo.d.ts +3 -0
  183. package/build/plugins/zenodo/package/convert/fromZenodo.js +43 -0
  184. package/build/plugins/zenodo/package/convert/toZenodo.d.ts +3 -0
  185. package/build/plugins/zenodo/package/convert/toZenodo.js +79 -0
  186. package/build/plugins/zenodo/package/index.d.ts +6 -0
  187. package/build/plugins/zenodo/package/index.js +5 -0
  188. package/build/plugins/zenodo/package/load.d.ts +23 -0
  189. package/build/plugins/zenodo/package/load.js +45 -0
  190. package/build/plugins/zenodo/package/load.spec.d.ts +1 -0
  191. package/build/plugins/zenodo/package/load.spec.js +15 -0
  192. package/build/plugins/zenodo/package/save.d.ts +13 -0
  193. package/build/plugins/zenodo/package/save.js +74 -0
  194. package/build/plugins/zenodo/package/save.spec.d.ts +1 -0
  195. package/build/plugins/zenodo/package/save.spec.js +524 -0
  196. package/build/plugins/zenodo/plugin.d.ts +19 -0
  197. package/build/plugins/zenodo/plugin.js +18 -0
  198. package/build/plugins/zenodo/plugin.spec.d.ts +1 -0
  199. package/build/plugins/zenodo/plugin.spec.js +78 -0
  200. package/build/plugins/zenodo/resource/Resource.d.ts +27 -0
  201. package/build/plugins/zenodo/resource/Resource.js +2 -0
  202. package/build/plugins/zenodo/resource/convert/fromZenodo.d.ts +10 -0
  203. package/build/plugins/zenodo/resource/convert/fromZenodo.js +18 -0
  204. package/build/plugins/zenodo/resource/convert/toZenodo.d.ts +3 -0
  205. package/build/plugins/zenodo/resource/convert/toZenodo.js +13 -0
  206. package/build/plugins/zenodo/resource/index.d.ts +3 -0
  207. package/build/plugins/zenodo/resource/index.js +3 -0
  208. package/build/plugins/zenodo/zenodo/index.d.ts +1 -0
  209. package/build/plugins/zenodo/zenodo/index.js +2 -0
  210. package/build/plugins/zenodo/zenodo/request.d.ts +12 -0
  211. package/build/plugins/zenodo/zenodo/request.js +36 -0
  212. package/build/plugins/zip/index.d.ts +2 -0
  213. package/build/plugins/zip/index.js +3 -0
  214. package/build/plugins/zip/package/index.d.ts +2 -0
  215. package/build/plugins/zip/package/index.js +3 -0
  216. package/build/plugins/zip/package/load.d.ts +1 -0
  217. package/build/plugins/zip/package/load.js +30 -0
  218. package/build/plugins/zip/package/load.spec.d.ts +1 -0
  219. package/build/plugins/zip/package/load.spec.js +173 -0
  220. package/build/plugins/zip/package/save.d.ts +5 -0
  221. package/build/plugins/zip/package/save.js +50 -0
  222. package/build/plugins/zip/package/save.spec.d.ts +1 -0
  223. package/build/plugins/zip/package/save.spec.js +287 -0
  224. package/build/plugins/zip/plugin.d.ts +11 -0
  225. package/build/plugins/zip/plugin.js +24 -0
  226. package/build/plugins/zip/plugin.spec.d.ts +1 -0
  227. package/build/plugins/zip/plugin.spec.js +158 -0
  228. package/build/resource/index.d.ts +1 -0
  229. package/build/resource/index.js +2 -0
  230. package/build/resource/save.d.ts +13 -0
  231. package/build/resource/save.js +53 -0
  232. package/build/resource/save.spec.d.ts +1 -0
  233. package/build/resource/save.spec.js +107 -0
  234. package/build/stream/concat.d.ts +3 -0
  235. package/build/stream/concat.js +5 -0
  236. package/build/stream/index.d.ts +3 -0
  237. package/build/stream/index.js +4 -0
  238. package/build/stream/load.d.ts +5 -0
  239. package/build/stream/load.js +46 -0
  240. package/build/stream/save.d.ts +5 -0
  241. package/build/stream/save.js +13 -0
  242. package/package.json +48 -0
@@ -0,0 +1,566 @@
1
+ import { relative } from "node:path";
2
+ import { loadPackageDescriptor } from "@frictionless-ts/metadata";
3
+ import { afterEach, beforeEach, describe, expect, it, vi } from "vitest";
4
+ import { savePackageToGithub } from "./save.js";
5
+ describe("savePackageToGithub", () => {
6
+ const getFixturePath = (name) => relative(process.cwd(), `${import.meta.dirname}/fixtures/${name}`);
7
+ const mockPackage = {
8
+ name: "test-package",
9
+ title: "Test Package",
10
+ description: "A test package",
11
+ version: "1.0.0",
12
+ resources: [
13
+ {
14
+ name: "test-resource",
15
+ path: getFixturePath("data.csv"),
16
+ format: "csv",
17
+ bytes: 100,
18
+ },
19
+ ],
20
+ };
21
+ const mockOptions = {
22
+ apiKey: "test-api-key",
23
+ repo: "test-repo",
24
+ };
25
+ const originalFetch = globalThis.fetch;
26
+ let fetchMock;
27
+ beforeEach(() => {
28
+ fetchMock = vi.fn();
29
+ // @ts-ignore
30
+ globalThis.fetch = fetchMock;
31
+ });
32
+ afterEach(() => {
33
+ globalThis.fetch = originalFetch;
34
+ vi.resetAllMocks();
35
+ });
36
+ it.skip("should save a package", async () => {
37
+ const dataPackage = await loadPackageDescriptor("core/package/fixtures/package.json");
38
+ const result = await savePackageToGithub(dataPackage, {
39
+ apiKey: "<key>",
40
+ repo: "test",
41
+ });
42
+ console.log(result);
43
+ expect(true).toBeDefined();
44
+ });
45
+ it("creates a repository in GitHub with correct API calls", async () => {
46
+ fetchMock.mockResolvedValueOnce({
47
+ ok: true,
48
+ json: () => Promise.resolve({
49
+ id: 12345,
50
+ name: "test-repo",
51
+ full_name: "test-user/test-repo",
52
+ owner: {
53
+ login: "test-user",
54
+ id: 1,
55
+ avatar_url: "https://avatars.githubusercontent.com/u/1",
56
+ html_url: "https://github.com/test-user",
57
+ type: "User",
58
+ },
59
+ html_url: "https://github.com/test-user/test-repo",
60
+ description: null,
61
+ created_at: "2024-01-01T00:00:00Z",
62
+ updated_at: "2024-01-01T00:00:00Z",
63
+ homepage: null,
64
+ size: 0,
65
+ stargazers_count: 0,
66
+ watchers_count: 0,
67
+ language: null,
68
+ license: null,
69
+ default_branch: "main",
70
+ topics: [],
71
+ private: false,
72
+ archived: false,
73
+ git_url: "git://github.com/test-user/test-repo.git",
74
+ ssh_url: "git@github.com:test-user/test-repo.git",
75
+ clone_url: "https://github.com/test-user/test-repo.git",
76
+ }),
77
+ });
78
+ fetchMock.mockResolvedValueOnce({
79
+ ok: true,
80
+ json: () => Promise.resolve({
81
+ content: {
82
+ name: "data.csv",
83
+ path: "data.csv",
84
+ sha: "abc123",
85
+ },
86
+ }),
87
+ });
88
+ fetchMock.mockResolvedValueOnce({
89
+ ok: true,
90
+ json: () => Promise.resolve({
91
+ content: {
92
+ name: "datapackage.json",
93
+ path: "datapackage.json",
94
+ sha: "def456",
95
+ },
96
+ }),
97
+ });
98
+ const result = await savePackageToGithub(mockPackage, mockOptions);
99
+ expect(fetchMock).toHaveBeenCalledTimes(3);
100
+ const repoCreateCall = fetchMock.mock.calls[0];
101
+ expect(repoCreateCall).toBeDefined();
102
+ if (!repoCreateCall)
103
+ return;
104
+ expect(repoCreateCall[0]).toEqual("https://api.github.com/user/repos");
105
+ expect(repoCreateCall[1]).toMatchObject({
106
+ method: "POST",
107
+ headers: {
108
+ Authorization: "Bearer test-api-key",
109
+ "Content-Type": "application/json",
110
+ },
111
+ });
112
+ const repoPayload = JSON.parse(repoCreateCall[1].body);
113
+ expect(repoPayload.name).toEqual("test-repo");
114
+ expect(repoPayload.auto_init).toEqual(true);
115
+ expect(result).toEqual({
116
+ path: "https://raw.githubusercontent.com/test-user/test-repo/refs/heads/main/dataPackage.json",
117
+ repoUrl: "https://github.com/test-user/test-repo",
118
+ });
119
+ });
120
+ it("creates a repository in an organization when org is specified", async () => {
121
+ fetchMock.mockResolvedValueOnce({
122
+ ok: true,
123
+ json: () => Promise.resolve({
124
+ id: 12345,
125
+ name: "test-repo",
126
+ full_name: "test-org/test-repo",
127
+ owner: {
128
+ login: "test-org",
129
+ id: 2,
130
+ avatar_url: "https://avatars.githubusercontent.com/u/2",
131
+ html_url: "https://github.com/test-org",
132
+ type: "Organization",
133
+ },
134
+ html_url: "https://github.com/test-org/test-repo",
135
+ description: null,
136
+ created_at: "2024-01-01T00:00:00Z",
137
+ updated_at: "2024-01-01T00:00:00Z",
138
+ homepage: null,
139
+ size: 0,
140
+ stargazers_count: 0,
141
+ watchers_count: 0,
142
+ language: null,
143
+ license: null,
144
+ default_branch: "main",
145
+ topics: [],
146
+ private: false,
147
+ archived: false,
148
+ git_url: "git://github.com/test-org/test-repo.git",
149
+ ssh_url: "git@github.com:test-org/test-repo.git",
150
+ clone_url: "https://github.com/test-org/test-repo.git",
151
+ }),
152
+ });
153
+ fetchMock.mockResolvedValue({
154
+ ok: true,
155
+ json: () => Promise.resolve({
156
+ content: {
157
+ name: "file",
158
+ path: "file",
159
+ sha: "abc123",
160
+ },
161
+ }),
162
+ });
163
+ await savePackageToGithub(mockPackage, {
164
+ ...mockOptions,
165
+ org: "test-org",
166
+ });
167
+ const repoCreateCall = fetchMock.mock.calls[0];
168
+ expect(repoCreateCall).toBeDefined();
169
+ if (!repoCreateCall)
170
+ return;
171
+ expect(repoCreateCall[0]).toEqual("https://api.github.com/orgs/test-org/repos");
172
+ });
173
+ it("uploads resource files with base64 encoding", async () => {
174
+ fetchMock.mockResolvedValueOnce({
175
+ ok: true,
176
+ json: () => Promise.resolve({
177
+ id: 12345,
178
+ name: "test-repo",
179
+ full_name: "test-user/test-repo",
180
+ owner: {
181
+ login: "test-user",
182
+ id: 1,
183
+ avatar_url: "https://avatars.githubusercontent.com/u/1",
184
+ html_url: "https://github.com/test-user",
185
+ type: "User",
186
+ },
187
+ html_url: "https://github.com/test-user/test-repo",
188
+ description: null,
189
+ created_at: "2024-01-01T00:00:00Z",
190
+ updated_at: "2024-01-01T00:00:00Z",
191
+ homepage: null,
192
+ size: 0,
193
+ stargazers_count: 0,
194
+ watchers_count: 0,
195
+ language: null,
196
+ license: null,
197
+ default_branch: "main",
198
+ topics: [],
199
+ private: false,
200
+ archived: false,
201
+ git_url: "git://github.com/test-user/test-repo.git",
202
+ ssh_url: "git@github.com:test-user/test-repo.git",
203
+ clone_url: "https://github.com/test-user/test-repo.git",
204
+ }),
205
+ });
206
+ fetchMock.mockResolvedValueOnce({
207
+ ok: true,
208
+ json: () => Promise.resolve({
209
+ content: {
210
+ name: "data.csv",
211
+ path: "data.csv",
212
+ sha: "abc123",
213
+ },
214
+ }),
215
+ });
216
+ fetchMock.mockResolvedValueOnce({
217
+ ok: true,
218
+ json: () => Promise.resolve({
219
+ content: {
220
+ name: "datapackage.json",
221
+ path: "datapackage.json",
222
+ sha: "def456",
223
+ },
224
+ }),
225
+ });
226
+ await savePackageToGithub(mockPackage, mockOptions);
227
+ const fileUploadCall = fetchMock.mock.calls[1];
228
+ expect(fileUploadCall).toBeDefined();
229
+ if (!fileUploadCall)
230
+ return;
231
+ expect(fileUploadCall[0]).toEqual("https://api.github.com/repos/test-user/test-repo/contents/data.csv");
232
+ expect(fileUploadCall[1]).toMatchObject({
233
+ method: "PUT",
234
+ headers: {
235
+ Authorization: "Bearer test-api-key",
236
+ "Content-Type": "application/json",
237
+ },
238
+ });
239
+ const filePayload = JSON.parse(fileUploadCall[1].body);
240
+ expect(filePayload.path).toEqual("data.csv");
241
+ expect(filePayload.message).toEqual('Added file "data.csv"');
242
+ expect(filePayload.content).toBeDefined();
243
+ expect(typeof filePayload.content).toEqual("string");
244
+ });
245
+ it("uploads datapackage.json metadata file", async () => {
246
+ fetchMock.mockResolvedValueOnce({
247
+ ok: true,
248
+ json: () => Promise.resolve({
249
+ id: 12345,
250
+ name: "test-repo",
251
+ full_name: "test-user/test-repo",
252
+ owner: {
253
+ login: "test-user",
254
+ id: 1,
255
+ avatar_url: "https://avatars.githubusercontent.com/u/1",
256
+ html_url: "https://github.com/test-user",
257
+ type: "User",
258
+ },
259
+ html_url: "https://github.com/test-user/test-repo",
260
+ description: null,
261
+ created_at: "2024-01-01T00:00:00Z",
262
+ updated_at: "2024-01-01T00:00:00Z",
263
+ homepage: null,
264
+ size: 0,
265
+ stargazers_count: 0,
266
+ watchers_count: 0,
267
+ language: null,
268
+ license: null,
269
+ default_branch: "main",
270
+ topics: [],
271
+ private: false,
272
+ archived: false,
273
+ git_url: "git://github.com/test-user/test-repo.git",
274
+ ssh_url: "git@github.com:test-user/test-repo.git",
275
+ clone_url: "https://github.com/test-user/test-repo.git",
276
+ }),
277
+ });
278
+ fetchMock.mockResolvedValueOnce({
279
+ ok: true,
280
+ json: () => Promise.resolve({
281
+ content: {
282
+ name: "data.csv",
283
+ path: "data.csv",
284
+ sha: "abc123",
285
+ },
286
+ }),
287
+ });
288
+ fetchMock.mockResolvedValueOnce({
289
+ ok: true,
290
+ json: () => Promise.resolve({
291
+ content: {
292
+ name: "datapackage.json",
293
+ path: "datapackage.json",
294
+ sha: "def456",
295
+ },
296
+ }),
297
+ });
298
+ await savePackageToGithub(mockPackage, mockOptions);
299
+ const datapackageUploadCall = fetchMock.mock.calls[2];
300
+ expect(datapackageUploadCall).toBeDefined();
301
+ if (!datapackageUploadCall)
302
+ return;
303
+ expect(datapackageUploadCall[0]).toEqual("https://api.github.com/repos/test-user/test-repo/contents/datapackage.json");
304
+ const datapackagePayload = JSON.parse(datapackageUploadCall[1].body);
305
+ expect(datapackagePayload.path).toEqual("datapackage.json");
306
+ expect(datapackagePayload.message).toEqual('Added file "datapackage.json"');
307
+ expect(datapackagePayload.content).toBeDefined();
308
+ expect(typeof datapackagePayload.content).toEqual("string");
309
+ });
310
+ it("passes API key as Bearer token in Authorization header", async () => {
311
+ fetchMock.mockResolvedValue({
312
+ ok: true,
313
+ json: () => Promise.resolve({
314
+ id: 12345,
315
+ name: "test-repo",
316
+ owner: {
317
+ login: "test-user",
318
+ id: 1,
319
+ avatar_url: "https://avatars.githubusercontent.com/u/1",
320
+ html_url: "https://github.com/test-user",
321
+ type: "User",
322
+ },
323
+ html_url: "https://github.com/test-user/test-repo",
324
+ description: null,
325
+ created_at: "2024-01-01T00:00:00Z",
326
+ updated_at: "2024-01-01T00:00:00Z",
327
+ homepage: null,
328
+ size: 0,
329
+ stargazers_count: 0,
330
+ watchers_count: 0,
331
+ language: null,
332
+ license: null,
333
+ default_branch: "main",
334
+ topics: [],
335
+ private: false,
336
+ archived: false,
337
+ git_url: "git://github.com/test-user/test-repo.git",
338
+ ssh_url: "git@github.com:test-user/test-repo.git",
339
+ clone_url: "https://github.com/test-user/test-repo.git",
340
+ }),
341
+ });
342
+ await savePackageToGithub(mockPackage, {
343
+ ...mockOptions,
344
+ apiKey: "custom-api-key",
345
+ });
346
+ const firstCall = fetchMock.mock.calls[0];
347
+ expect(firstCall).toBeDefined();
348
+ if (!firstCall)
349
+ return;
350
+ const headers = firstCall[1].headers;
351
+ expect(headers.Authorization).toEqual("Bearer custom-api-key");
352
+ });
353
+ it("handles API errors from repository creation", async () => {
354
+ fetchMock.mockResolvedValueOnce({
355
+ ok: false,
356
+ status: 400,
357
+ statusText: "Bad Request",
358
+ text: () => Promise.resolve("Repository name already exists"),
359
+ });
360
+ await expect(savePackageToGithub(mockPackage, mockOptions)).rejects.toThrow("Github API error: 400 Bad Request");
361
+ });
362
+ it("handles API errors from file upload", async () => {
363
+ fetchMock.mockResolvedValueOnce({
364
+ ok: true,
365
+ json: () => Promise.resolve({
366
+ id: 12345,
367
+ name: "test-repo",
368
+ full_name: "test-user/test-repo",
369
+ owner: {
370
+ login: "test-user",
371
+ id: 1,
372
+ avatar_url: "https://avatars.githubusercontent.com/u/1",
373
+ html_url: "https://github.com/test-user",
374
+ type: "User",
375
+ },
376
+ html_url: "https://github.com/test-user/test-repo",
377
+ description: null,
378
+ created_at: "2024-01-01T00:00:00Z",
379
+ updated_at: "2024-01-01T00:00:00Z",
380
+ homepage: null,
381
+ size: 0,
382
+ stargazers_count: 0,
383
+ watchers_count: 0,
384
+ language: null,
385
+ license: null,
386
+ default_branch: "main",
387
+ topics: [],
388
+ private: false,
389
+ archived: false,
390
+ git_url: "git://github.com/test-user/test-repo.git",
391
+ ssh_url: "git@github.com:test-user/test-repo.git",
392
+ clone_url: "https://github.com/test-user/test-repo.git",
393
+ }),
394
+ });
395
+ fetchMock.mockResolvedValueOnce({
396
+ ok: false,
397
+ status: 500,
398
+ statusText: "Internal Server Error",
399
+ text: () => Promise.resolve("Failed to upload file"),
400
+ });
401
+ await expect(savePackageToGithub(mockPackage, mockOptions)).rejects.toThrow("Github API error: 500 Internal Server Error");
402
+ });
403
+ it("handles packages with multiple resources", async () => {
404
+ const multiResourcePackage = {
405
+ ...mockPackage,
406
+ resources: [
407
+ {
408
+ name: "resource-1",
409
+ path: getFixturePath("data.csv"),
410
+ format: "csv",
411
+ },
412
+ {
413
+ name: "resource-2",
414
+ path: getFixturePath("data.csv"),
415
+ format: "json",
416
+ },
417
+ ],
418
+ };
419
+ fetchMock.mockResolvedValue({
420
+ ok: true,
421
+ json: () => Promise.resolve({
422
+ id: 12345,
423
+ name: "test-repo",
424
+ full_name: "test-user/test-repo",
425
+ owner: {
426
+ login: "test-user",
427
+ id: 1,
428
+ avatar_url: "https://avatars.githubusercontent.com/u/1",
429
+ html_url: "https://github.com/test-user",
430
+ type: "User",
431
+ },
432
+ html_url: "https://github.com/test-user/test-repo",
433
+ description: null,
434
+ created_at: "2024-01-01T00:00:00Z",
435
+ updated_at: "2024-01-01T00:00:00Z",
436
+ homepage: null,
437
+ size: 0,
438
+ stargazers_count: 0,
439
+ watchers_count: 0,
440
+ language: null,
441
+ license: null,
442
+ default_branch: "main",
443
+ topics: [],
444
+ private: false,
445
+ archived: false,
446
+ git_url: "git://github.com/test-user/test-repo.git",
447
+ ssh_url: "git@github.com:test-user/test-repo.git",
448
+ clone_url: "https://github.com/test-user/test-repo.git",
449
+ }),
450
+ });
451
+ await savePackageToGithub(multiResourcePackage, mockOptions);
452
+ expect(fetchMock).toHaveBeenCalledTimes(4);
453
+ const secondFileUploadCall = fetchMock.mock.calls[2];
454
+ expect(secondFileUploadCall).toBeDefined();
455
+ if (!secondFileUploadCall)
456
+ return;
457
+ expect(secondFileUploadCall[0]).toContain("/contents/");
458
+ });
459
+ it("handles packages with no resources", async () => {
460
+ const emptyPackage = {
461
+ ...mockPackage,
462
+ resources: [],
463
+ };
464
+ fetchMock.mockResolvedValueOnce({
465
+ ok: true,
466
+ json: () => Promise.resolve({
467
+ id: 12345,
468
+ name: "test-repo",
469
+ full_name: "test-user/test-repo",
470
+ owner: {
471
+ login: "test-user",
472
+ id: 1,
473
+ avatar_url: "https://avatars.githubusercontent.com/u/1",
474
+ html_url: "https://github.com/test-user",
475
+ type: "User",
476
+ },
477
+ html_url: "https://github.com/test-user/test-repo",
478
+ description: null,
479
+ created_at: "2024-01-01T00:00:00Z",
480
+ updated_at: "2024-01-01T00:00:00Z",
481
+ homepage: null,
482
+ size: 0,
483
+ stargazers_count: 0,
484
+ watchers_count: 0,
485
+ language: null,
486
+ license: null,
487
+ default_branch: "main",
488
+ topics: [],
489
+ private: false,
490
+ archived: false,
491
+ git_url: "git://github.com/test-user/test-repo.git",
492
+ ssh_url: "git@github.com:test-user/test-repo.git",
493
+ clone_url: "https://github.com/test-user/test-repo.git",
494
+ }),
495
+ });
496
+ fetchMock.mockResolvedValueOnce({
497
+ ok: true,
498
+ json: () => Promise.resolve({
499
+ content: {
500
+ name: "datapackage.json",
501
+ path: "datapackage.json",
502
+ sha: "def456",
503
+ },
504
+ }),
505
+ });
506
+ const result = await savePackageToGithub(emptyPackage, mockOptions);
507
+ expect(fetchMock).toHaveBeenCalledTimes(2);
508
+ expect(result.repoUrl).toEqual("https://github.com/test-user/test-repo");
509
+ });
510
+ it("skips resources without path", async () => {
511
+ const packageWithoutPath = {
512
+ ...mockPackage,
513
+ resources: [
514
+ {
515
+ name: "resource-without-path",
516
+ format: "csv",
517
+ },
518
+ ],
519
+ };
520
+ fetchMock.mockResolvedValueOnce({
521
+ ok: true,
522
+ json: () => Promise.resolve({
523
+ id: 12345,
524
+ name: "test-repo",
525
+ full_name: "test-user/test-repo",
526
+ owner: {
527
+ login: "test-user",
528
+ id: 1,
529
+ avatar_url: "https://avatars.githubusercontent.com/u/1",
530
+ html_url: "https://github.com/test-user",
531
+ type: "User",
532
+ },
533
+ html_url: "https://github.com/test-user/test-repo",
534
+ description: null,
535
+ created_at: "2024-01-01T00:00:00Z",
536
+ updated_at: "2024-01-01T00:00:00Z",
537
+ homepage: null,
538
+ size: 0,
539
+ stargazers_count: 0,
540
+ watchers_count: 0,
541
+ language: null,
542
+ license: null,
543
+ default_branch: "main",
544
+ topics: [],
545
+ private: false,
546
+ archived: false,
547
+ git_url: "git://github.com/test-user/test-repo.git",
548
+ ssh_url: "git@github.com:test-user/test-repo.git",
549
+ clone_url: "https://github.com/test-user/test-repo.git",
550
+ }),
551
+ });
552
+ fetchMock.mockResolvedValueOnce({
553
+ ok: true,
554
+ json: () => Promise.resolve({
555
+ content: {
556
+ name: "datapackage.json",
557
+ path: "datapackage.json",
558
+ sha: "def456",
559
+ },
560
+ }),
561
+ });
562
+ await savePackageToGithub(packageWithoutPath, mockOptions);
563
+ expect(fetchMock).toHaveBeenCalledTimes(2);
564
+ });
565
+ });
566
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"save.spec.js","sourceRoot":"","sources":["../../../../plugins/github/package/save.spec.ts"],"names":[],"mappings":"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","sourcesContent":["import { relative } from \"node:path\"\nimport type { Package } from \"@frictionless-ts/metadata\"\nimport { loadPackageDescriptor } from \"@frictionless-ts/metadata\"\nimport { afterEach, beforeEach, describe, expect, it, vi } from \"vitest\"\nimport { savePackageToGithub } from \"./save.ts\"\n\ndescribe(\"savePackageToGithub\", () => {\n  const getFixturePath = (name: string) =>\n    relative(process.cwd(), `${import.meta.dirname}/fixtures/${name}`)\n\n  const mockPackage: Package = {\n    name: \"test-package\",\n    title: \"Test Package\",\n    description: \"A test package\",\n    version: \"1.0.0\",\n    resources: [\n      {\n        name: \"test-resource\",\n        path: getFixturePath(\"data.csv\"),\n        format: \"csv\",\n        bytes: 100,\n      },\n    ],\n  }\n\n  const mockOptions = {\n    apiKey: \"test-api-key\",\n    repo: \"test-repo\",\n  }\n\n  const originalFetch = globalThis.fetch\n  let fetchMock: ReturnType<typeof vi.fn>\n\n  beforeEach(() => {\n    fetchMock = vi.fn()\n    // @ts-ignore\n    globalThis.fetch = fetchMock\n  })\n\n  afterEach(() => {\n    globalThis.fetch = originalFetch\n    vi.resetAllMocks()\n  })\n\n  it.skip(\"should save a package\", async () => {\n    const dataPackage = await loadPackageDescriptor(\n      \"core/package/fixtures/package.json\",\n    )\n\n    const result = await savePackageToGithub(dataPackage, {\n      apiKey: \"<key>\",\n      repo: \"test\",\n    })\n\n    console.log(result)\n\n    expect(true).toBeDefined()\n  })\n\n  it(\"creates a repository in GitHub with correct API calls\", async () => {\n    fetchMock.mockResolvedValueOnce({\n      ok: true,\n      json: () =>\n        Promise.resolve({\n          id: 12345,\n          name: \"test-repo\",\n          full_name: \"test-user/test-repo\",\n          owner: {\n            login: \"test-user\",\n            id: 1,\n            avatar_url: \"https://avatars.githubusercontent.com/u/1\",\n            html_url: \"https://github.com/test-user\",\n            type: \"User\",\n          },\n          html_url: \"https://github.com/test-user/test-repo\",\n          description: null,\n          created_at: \"2024-01-01T00:00:00Z\",\n          updated_at: \"2024-01-01T00:00:00Z\",\n          homepage: null,\n          size: 0,\n          stargazers_count: 0,\n          watchers_count: 0,\n          language: null,\n          license: null,\n          default_branch: \"main\",\n          topics: [],\n          private: false,\n          archived: false,\n          git_url: \"git://github.com/test-user/test-repo.git\",\n          ssh_url: \"git@github.com:test-user/test-repo.git\",\n          clone_url: \"https://github.com/test-user/test-repo.git\",\n        }),\n    })\n\n    fetchMock.mockResolvedValueOnce({\n      ok: true,\n      json: () =>\n        Promise.resolve({\n          content: {\n            name: \"data.csv\",\n            path: \"data.csv\",\n            sha: \"abc123\",\n          },\n        }),\n    })\n\n    fetchMock.mockResolvedValueOnce({\n      ok: true,\n      json: () =>\n        Promise.resolve({\n          content: {\n            name: \"datapackage.json\",\n            path: \"datapackage.json\",\n            sha: \"def456\",\n          },\n        }),\n    })\n\n    const result = await savePackageToGithub(mockPackage, mockOptions)\n\n    expect(fetchMock).toHaveBeenCalledTimes(3)\n\n    const repoCreateCall = fetchMock.mock.calls[0]\n    expect(repoCreateCall).toBeDefined()\n    if (!repoCreateCall) return\n\n    expect(repoCreateCall[0]).toEqual(\"https://api.github.com/user/repos\")\n    expect(repoCreateCall[1]).toMatchObject({\n      method: \"POST\",\n      headers: {\n        Authorization: \"Bearer test-api-key\",\n        \"Content-Type\": \"application/json\",\n      },\n    })\n\n    const repoPayload = JSON.parse(repoCreateCall[1].body)\n    expect(repoPayload.name).toEqual(\"test-repo\")\n    expect(repoPayload.auto_init).toEqual(true)\n\n    expect(result).toEqual({\n      path: \"https://raw.githubusercontent.com/test-user/test-repo/refs/heads/main/dataPackage.json\",\n      repoUrl: \"https://github.com/test-user/test-repo\",\n    })\n  })\n\n  it(\"creates a repository in an organization when org is specified\", async () => {\n    fetchMock.mockResolvedValueOnce({\n      ok: true,\n      json: () =>\n        Promise.resolve({\n          id: 12345,\n          name: \"test-repo\",\n          full_name: \"test-org/test-repo\",\n          owner: {\n            login: \"test-org\",\n            id: 2,\n            avatar_url: \"https://avatars.githubusercontent.com/u/2\",\n            html_url: \"https://github.com/test-org\",\n            type: \"Organization\",\n          },\n          html_url: \"https://github.com/test-org/test-repo\",\n          description: null,\n          created_at: \"2024-01-01T00:00:00Z\",\n          updated_at: \"2024-01-01T00:00:00Z\",\n          homepage: null,\n          size: 0,\n          stargazers_count: 0,\n          watchers_count: 0,\n          language: null,\n          license: null,\n          default_branch: \"main\",\n          topics: [],\n          private: false,\n          archived: false,\n          git_url: \"git://github.com/test-org/test-repo.git\",\n          ssh_url: \"git@github.com:test-org/test-repo.git\",\n          clone_url: \"https://github.com/test-org/test-repo.git\",\n        }),\n    })\n\n    fetchMock.mockResolvedValue({\n      ok: true,\n      json: () =>\n        Promise.resolve({\n          content: {\n            name: \"file\",\n            path: \"file\",\n            sha: \"abc123\",\n          },\n        }),\n    })\n\n    await savePackageToGithub(mockPackage, {\n      ...mockOptions,\n      org: \"test-org\",\n    })\n\n    const repoCreateCall = fetchMock.mock.calls[0]\n    expect(repoCreateCall).toBeDefined()\n    if (!repoCreateCall) return\n\n    expect(repoCreateCall[0]).toEqual(\n      \"https://api.github.com/orgs/test-org/repos\",\n    )\n  })\n\n  it(\"uploads resource files with base64 encoding\", async () => {\n    fetchMock.mockResolvedValueOnce({\n      ok: true,\n      json: () =>\n        Promise.resolve({\n          id: 12345,\n          name: \"test-repo\",\n          full_name: \"test-user/test-repo\",\n          owner: {\n            login: \"test-user\",\n            id: 1,\n            avatar_url: \"https://avatars.githubusercontent.com/u/1\",\n            html_url: \"https://github.com/test-user\",\n            type: \"User\",\n          },\n          html_url: \"https://github.com/test-user/test-repo\",\n          description: null,\n          created_at: \"2024-01-01T00:00:00Z\",\n          updated_at: \"2024-01-01T00:00:00Z\",\n          homepage: null,\n          size: 0,\n          stargazers_count: 0,\n          watchers_count: 0,\n          language: null,\n          license: null,\n          default_branch: \"main\",\n          topics: [],\n          private: false,\n          archived: false,\n          git_url: \"git://github.com/test-user/test-repo.git\",\n          ssh_url: \"git@github.com:test-user/test-repo.git\",\n          clone_url: \"https://github.com/test-user/test-repo.git\",\n        }),\n    })\n\n    fetchMock.mockResolvedValueOnce({\n      ok: true,\n      json: () =>\n        Promise.resolve({\n          content: {\n            name: \"data.csv\",\n            path: \"data.csv\",\n            sha: \"abc123\",\n          },\n        }),\n    })\n\n    fetchMock.mockResolvedValueOnce({\n      ok: true,\n      json: () =>\n        Promise.resolve({\n          content: {\n            name: \"datapackage.json\",\n            path: \"datapackage.json\",\n            sha: \"def456\",\n          },\n        }),\n    })\n\n    await savePackageToGithub(mockPackage, mockOptions)\n\n    const fileUploadCall = fetchMock.mock.calls[1]\n    expect(fileUploadCall).toBeDefined()\n    if (!fileUploadCall) return\n\n    expect(fileUploadCall[0]).toEqual(\n      \"https://api.github.com/repos/test-user/test-repo/contents/data.csv\",\n    )\n    expect(fileUploadCall[1]).toMatchObject({\n      method: \"PUT\",\n      headers: {\n        Authorization: \"Bearer test-api-key\",\n        \"Content-Type\": \"application/json\",\n      },\n    })\n\n    const filePayload = JSON.parse(fileUploadCall[1].body)\n    expect(filePayload.path).toEqual(\"data.csv\")\n    expect(filePayload.message).toEqual('Added file \"data.csv\"')\n    expect(filePayload.content).toBeDefined()\n    expect(typeof filePayload.content).toEqual(\"string\")\n  })\n\n  it(\"uploads datapackage.json metadata file\", async () => {\n    fetchMock.mockResolvedValueOnce({\n      ok: true,\n      json: () =>\n        Promise.resolve({\n          id: 12345,\n          name: \"test-repo\",\n          full_name: \"test-user/test-repo\",\n          owner: {\n            login: \"test-user\",\n            id: 1,\n            avatar_url: \"https://avatars.githubusercontent.com/u/1\",\n            html_url: \"https://github.com/test-user\",\n            type: \"User\",\n          },\n          html_url: \"https://github.com/test-user/test-repo\",\n          description: null,\n          created_at: \"2024-01-01T00:00:00Z\",\n          updated_at: \"2024-01-01T00:00:00Z\",\n          homepage: null,\n          size: 0,\n          stargazers_count: 0,\n          watchers_count: 0,\n          language: null,\n          license: null,\n          default_branch: \"main\",\n          topics: [],\n          private: false,\n          archived: false,\n          git_url: \"git://github.com/test-user/test-repo.git\",\n          ssh_url: \"git@github.com:test-user/test-repo.git\",\n          clone_url: \"https://github.com/test-user/test-repo.git\",\n        }),\n    })\n\n    fetchMock.mockResolvedValueOnce({\n      ok: true,\n      json: () =>\n        Promise.resolve({\n          content: {\n            name: \"data.csv\",\n            path: \"data.csv\",\n            sha: \"abc123\",\n          },\n        }),\n    })\n\n    fetchMock.mockResolvedValueOnce({\n      ok: true,\n      json: () =>\n        Promise.resolve({\n          content: {\n            name: \"datapackage.json\",\n            path: \"datapackage.json\",\n            sha: \"def456\",\n          },\n        }),\n    })\n\n    await savePackageToGithub(mockPackage, mockOptions)\n\n    const datapackageUploadCall = fetchMock.mock.calls[2]\n    expect(datapackageUploadCall).toBeDefined()\n    if (!datapackageUploadCall) return\n\n    expect(datapackageUploadCall[0]).toEqual(\n      \"https://api.github.com/repos/test-user/test-repo/contents/datapackage.json\",\n    )\n\n    const datapackagePayload = JSON.parse(datapackageUploadCall[1].body)\n    expect(datapackagePayload.path).toEqual(\"datapackage.json\")\n    expect(datapackagePayload.message).toEqual('Added file \"datapackage.json\"')\n    expect(datapackagePayload.content).toBeDefined()\n    expect(typeof datapackagePayload.content).toEqual(\"string\")\n  })\n\n  it(\"passes API key as Bearer token in Authorization header\", async () => {\n    fetchMock.mockResolvedValue({\n      ok: true,\n      json: () =>\n        Promise.resolve({\n          id: 12345,\n          name: \"test-repo\",\n          owner: {\n            login: \"test-user\",\n            id: 1,\n            avatar_url: \"https://avatars.githubusercontent.com/u/1\",\n            html_url: \"https://github.com/test-user\",\n            type: \"User\",\n          },\n          html_url: \"https://github.com/test-user/test-repo\",\n          description: null,\n          created_at: \"2024-01-01T00:00:00Z\",\n          updated_at: \"2024-01-01T00:00:00Z\",\n          homepage: null,\n          size: 0,\n          stargazers_count: 0,\n          watchers_count: 0,\n          language: null,\n          license: null,\n          default_branch: \"main\",\n          topics: [],\n          private: false,\n          archived: false,\n          git_url: \"git://github.com/test-user/test-repo.git\",\n          ssh_url: \"git@github.com:test-user/test-repo.git\",\n          clone_url: \"https://github.com/test-user/test-repo.git\",\n        }),\n    })\n\n    await savePackageToGithub(mockPackage, {\n      ...mockOptions,\n      apiKey: \"custom-api-key\",\n    })\n\n    const firstCall = fetchMock.mock.calls[0]\n    expect(firstCall).toBeDefined()\n    if (!firstCall) return\n\n    const headers = firstCall[1].headers\n    expect(headers.Authorization).toEqual(\"Bearer custom-api-key\")\n  })\n\n  it(\"handles API errors from repository creation\", async () => {\n    fetchMock.mockResolvedValueOnce({\n      ok: false,\n      status: 400,\n      statusText: \"Bad Request\",\n      text: () => Promise.resolve(\"Repository name already exists\"),\n    })\n\n    await expect(savePackageToGithub(mockPackage, mockOptions)).rejects.toThrow(\n      \"Github API error: 400 Bad Request\",\n    )\n  })\n\n  it(\"handles API errors from file upload\", async () => {\n    fetchMock.mockResolvedValueOnce({\n      ok: true,\n      json: () =>\n        Promise.resolve({\n          id: 12345,\n          name: \"test-repo\",\n          full_name: \"test-user/test-repo\",\n          owner: {\n            login: \"test-user\",\n            id: 1,\n            avatar_url: \"https://avatars.githubusercontent.com/u/1\",\n            html_url: \"https://github.com/test-user\",\n            type: \"User\",\n          },\n          html_url: \"https://github.com/test-user/test-repo\",\n          description: null,\n          created_at: \"2024-01-01T00:00:00Z\",\n          updated_at: \"2024-01-01T00:00:00Z\",\n          homepage: null,\n          size: 0,\n          stargazers_count: 0,\n          watchers_count: 0,\n          language: null,\n          license: null,\n          default_branch: \"main\",\n          topics: [],\n          private: false,\n          archived: false,\n          git_url: \"git://github.com/test-user/test-repo.git\",\n          ssh_url: \"git@github.com:test-user/test-repo.git\",\n          clone_url: \"https://github.com/test-user/test-repo.git\",\n        }),\n    })\n\n    fetchMock.mockResolvedValueOnce({\n      ok: false,\n      status: 500,\n      statusText: \"Internal Server Error\",\n      text: () => Promise.resolve(\"Failed to upload file\"),\n    })\n\n    await expect(savePackageToGithub(mockPackage, mockOptions)).rejects.toThrow(\n      \"Github API error: 500 Internal Server Error\",\n    )\n  })\n\n  it(\"handles packages with multiple resources\", async () => {\n    const multiResourcePackage: Package = {\n      ...mockPackage,\n      resources: [\n        {\n          name: \"resource-1\",\n          path: getFixturePath(\"data.csv\"),\n          format: \"csv\",\n        },\n        {\n          name: \"resource-2\",\n          path: getFixturePath(\"data.csv\"),\n          format: \"json\",\n        },\n      ],\n    }\n\n    fetchMock.mockResolvedValue({\n      ok: true,\n      json: () =>\n        Promise.resolve({\n          id: 12345,\n          name: \"test-repo\",\n          full_name: \"test-user/test-repo\",\n          owner: {\n            login: \"test-user\",\n            id: 1,\n            avatar_url: \"https://avatars.githubusercontent.com/u/1\",\n            html_url: \"https://github.com/test-user\",\n            type: \"User\",\n          },\n          html_url: \"https://github.com/test-user/test-repo\",\n          description: null,\n          created_at: \"2024-01-01T00:00:00Z\",\n          updated_at: \"2024-01-01T00:00:00Z\",\n          homepage: null,\n          size: 0,\n          stargazers_count: 0,\n          watchers_count: 0,\n          language: null,\n          license: null,\n          default_branch: \"main\",\n          topics: [],\n          private: false,\n          archived: false,\n          git_url: \"git://github.com/test-user/test-repo.git\",\n          ssh_url: \"git@github.com:test-user/test-repo.git\",\n          clone_url: \"https://github.com/test-user/test-repo.git\",\n        }),\n    })\n\n    await savePackageToGithub(multiResourcePackage, mockOptions)\n\n    expect(fetchMock).toHaveBeenCalledTimes(4)\n\n    const secondFileUploadCall = fetchMock.mock.calls[2]\n    expect(secondFileUploadCall).toBeDefined()\n    if (!secondFileUploadCall) return\n\n    expect(secondFileUploadCall[0]).toContain(\"/contents/\")\n  })\n\n  it(\"handles packages with no resources\", async () => {\n    const emptyPackage: Package = {\n      ...mockPackage,\n      resources: [],\n    }\n\n    fetchMock.mockResolvedValueOnce({\n      ok: true,\n      json: () =>\n        Promise.resolve({\n          id: 12345,\n          name: \"test-repo\",\n          full_name: \"test-user/test-repo\",\n          owner: {\n            login: \"test-user\",\n            id: 1,\n            avatar_url: \"https://avatars.githubusercontent.com/u/1\",\n            html_url: \"https://github.com/test-user\",\n            type: \"User\",\n          },\n          html_url: \"https://github.com/test-user/test-repo\",\n          description: null,\n          created_at: \"2024-01-01T00:00:00Z\",\n          updated_at: \"2024-01-01T00:00:00Z\",\n          homepage: null,\n          size: 0,\n          stargazers_count: 0,\n          watchers_count: 0,\n          language: null,\n          license: null,\n          default_branch: \"main\",\n          topics: [],\n          private: false,\n          archived: false,\n          git_url: \"git://github.com/test-user/test-repo.git\",\n          ssh_url: \"git@github.com:test-user/test-repo.git\",\n          clone_url: \"https://github.com/test-user/test-repo.git\",\n        }),\n    })\n\n    fetchMock.mockResolvedValueOnce({\n      ok: true,\n      json: () =>\n        Promise.resolve({\n          content: {\n            name: \"datapackage.json\",\n            path: \"datapackage.json\",\n            sha: \"def456\",\n          },\n        }),\n    })\n\n    const result = await savePackageToGithub(emptyPackage, mockOptions)\n\n    expect(fetchMock).toHaveBeenCalledTimes(2)\n    expect(result.repoUrl).toEqual(\"https://github.com/test-user/test-repo\")\n  })\n\n  it(\"skips resources without path\", async () => {\n    const packageWithoutPath: Package = {\n      ...mockPackage,\n      resources: [\n        {\n          name: \"resource-without-path\",\n          format: \"csv\",\n        },\n      ],\n    }\n\n    fetchMock.mockResolvedValueOnce({\n      ok: true,\n      json: () =>\n        Promise.resolve({\n          id: 12345,\n          name: \"test-repo\",\n          full_name: \"test-user/test-repo\",\n          owner: {\n            login: \"test-user\",\n            id: 1,\n            avatar_url: \"https://avatars.githubusercontent.com/u/1\",\n            html_url: \"https://github.com/test-user\",\n            type: \"User\",\n          },\n          html_url: \"https://github.com/test-user/test-repo\",\n          description: null,\n          created_at: \"2024-01-01T00:00:00Z\",\n          updated_at: \"2024-01-01T00:00:00Z\",\n          homepage: null,\n          size: 0,\n          stargazers_count: 0,\n          watchers_count: 0,\n          language: null,\n          license: null,\n          default_branch: \"main\",\n          topics: [],\n          private: false,\n          archived: false,\n          git_url: \"git://github.com/test-user/test-repo.git\",\n          ssh_url: \"git@github.com:test-user/test-repo.git\",\n          clone_url: \"https://github.com/test-user/test-repo.git\",\n        }),\n    })\n\n    fetchMock.mockResolvedValueOnce({\n      ok: true,\n      json: () =>\n        Promise.resolve({\n          content: {\n            name: \"datapackage.json\",\n            path: \"datapackage.json\",\n            sha: \"def456\",\n          },\n        }),\n    })\n\n    await savePackageToGithub(packageWithoutPath, mockOptions)\n\n    expect(fetchMock).toHaveBeenCalledTimes(2)\n  })\n})\n"]}