-
Notifications
You must be signed in to change notification settings - Fork 29
/
Copy pathutils.js
executable file
·209 lines (185 loc) · 7.14 KB
/
utils.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
// Common utility functions
// Helper to get server URL with fallback to legacy config name
export async function getArchiveBoxServerUrl() {
const { archivebox_server_url } = await chrome.storage.local.get(['archivebox_server_url']); // new ArchiveBox Extension v2.1.3 location
const {config_archiveBoxBaseUrl} = await chrome.storage.sync.get(['config_archiveBoxBaseUrl']); // old ArchiveBox Exporter v1.3.1 location
return archivebox_server_url || config_archiveBoxBaseUrl || '';
}
export function filterEntries(entries, filterText) {
if (!filterText) return entries;
const searchTerms = filterText.toLowerCase().split(' ');
return entries.filter(entry => {
const searchableText = [
entry.url,
entry.title,
entry.id,
new Date(entry.timestamp).toISOString(),
...entry.tags
].join(' ').toLowerCase();
return searchTerms.every(term => searchableText.includes(term));
});
}
// Common function to format cookies for export used in both personas-tab.js and cookies-tab.js
export function formatCookiesForExport(cookies) {
return Object.entries(cookies).map(([domain, domainCookies]) => {
return `# ${domain}\n${domainCookies.map(cookie =>
`${cookie.name}=${cookie.value}; domain=${cookie.domain}; path=${cookie.path}`
).join('\n')}`;
}).join('\n\n');
}
// Status indicator update helper
export function updateStatusIndicator(indicator, textElement, success, message) {
indicator.className = success ? 'status-indicator status-success' : 'status-indicator status-error';
textElement.textContent = message;
textElement.className = success ? 'text-success' : 'text-danger';
}
export async function addToArchiveBox(addCommandArgs, onComplete, onError) {
console.log('i addToArchiveBox', addCommandArgs);
try {
const archivebox_server_url = await getArchiveBoxServerUrl();
const { archivebox_api_key } = await chrome.storage.local.get(['archivebox_api_key']);
console.log('i addToArchiveBox server url', archivebox_server_url);
if (!archivebox_server_url) {
throw new Error('Server not configured.');
}
if (archivebox_api_key) {
// try ArchiveBox v0.8.0+ API endpoint first
try {
const response = await fetch(`${archivebox_server_url}/api/v1/cli/add`, {
headers: {
'x-archivebox-api-key': `${archivebox_api_key}`
},
method: 'post',
credentials: 'include',
body: addCommandArgs
});
if (response.ok) {
console.log('i addToArchiveBox using v0.8.5 REST API succeeded', response.status, response.statusText);
onComplete({ok: response.ok, status: response.status, statusText: response.statusText});
return true;
} else {
console.warn(`! addToArchiveBox using v0.8.5 REST API failed with status ${response.status} ${response.statusText}`);
// Fall through to legacy API
}
} catch (error) {
console.warn('! addToArchiveBox using v0.8.5 REST API failed with error:', error.message);
// Fall through to legacy API
}
}
// fall back to pre-v0.8.0 endpoint for backwards compatibility
console.log('i addToArchiveBox using legacy /add POST method');
const parsedAddCommandArgs = JSON.parse(addCommandArgs);
const urls = parsedAddCommandArgs && parsedAddCommandArgs.urls
? parsedAddCommandArgs.urls.join("\n") : "";
const tags = parsedAddCommandArgs && parsedAddCommandArgs.tags
? parsedAddCommandArgs.tags : "";
const body = new FormData();
body.append("url", urls);
body.append("tag", tags);
body.append("parser", "auto")
body.append("depth", 0)
try {
const response = await fetch(`${archivebox_server_url}/add/`, {
method: "post",
credentials: "include",
body: body
});
if (response.ok) {
console.log('i addToArchiveBox using legacy /add POST method succeeded', response.status, response.statusText);
onComplete({ok: response.ok, status: response.status, statusText: response.statusText});
} else {
console.error(`! addToArchiveBox using legacy /add POST method failed: ${response.status} ${response.statusText}`);
onError({ok: false, errorMessage: `HTTP ${response.status}: ${response.statusText}`});
}
} catch (error) {
console.error('! addToArchiveBox using legacy /add POST method failed with error:', error.message);
onError({ok: false, errorMessage: error.message});
}
} catch (e) {
console.error('! addToArchiveBox failed', e.message);
onError({ok: false, errorMessage: e.message});
}
return true;
}
export function downloadCsv(entries) {
const headers = ['id', 'timestamp', 'url', 'title', 'tags', 'notes'];
const csvRows = [
headers.join(','),
...entries.map(entry => {
return [
entry.id,
entry.timestamp,
`"${entry.url}"`,
`"${entry.title || ''}"`,
`"${entry.tags.join(';')}"`,
`"${entry.notes || ''}"`
].join(',');
})
];
const csvContent = csvRows.join('\n');
const blob = new Blob([csvContent], { type: 'text/csv;charset=utf-8;' });
const url = URL.createObjectURL(blob);
const link = document.createElement('a');
link.setAttribute('href', url);
link.setAttribute('download', `archivebox-export-${new Date().toISOString().split('T')[0]}.csv`);
document.body.appendChild(link);
link.click();
document.body.removeChild(link);
}
export function downloadJson(entries) {
const jsonContent = JSON.stringify(entries, null, 2);
const blob = new Blob([jsonContent], { type: 'application/json;charset=utf-8;' });
const url = URL.createObjectURL(blob);
const link = document.createElement('a');
link.setAttribute('href', url);
link.setAttribute('download', `archivebox-export-${new Date().toISOString().split('T')[0]}.json`);
document.body.appendChild(link);
link.click();
document.body.removeChild(link);
}
// Shared syncToArchiveBox function for config-tab and entries-tab
export async function syncToArchiveBox(entry) {
const archivebox_server_url = await getArchiveBoxServerUrl();
const { archivebox_api_key } = await chrome.storage.local.get(['archivebox_api_key']);
if (!archivebox_server_url || !archivebox_api_key) {
return {
ok: false,
status: 'Server URL and API key must be configured and saved first'
};
}
try {
const response = await fetch(`${archivebox_server_url}/api/v1/cli/add`, {
method: 'POST',
mode: 'cors',
credentials: 'omit',
headers: {
'Accept': 'application/json',
'Content-Type': 'application/json',
'x-archivebox-api-key': archivebox_api_key,
},
body: JSON.stringify({
urls: [entry.url],
tag: entry.tags.join(','),
depth: 0,
update: false,
update_all: false,
}),
});
if (!response.ok) {
const text = await response.text();
return {
ok: false,
status: `Server returned ${response.status}: ${text}`
};
}
return {
ok: true,
status: 'Success'
};
} catch (err) {
return {
ok: false,
status: `Connection failed: ${err.message}`
};
}
}