Extract extension data from Discord posts or GitHub repos and copy as formatted JSON (Alt+Shift+E)
Size
28.9 KB
Version
1.1.2
Created
Oct 18, 2025
Updated
5 days ago
1// ==UserScript==
2// @name SillyTavern Extension Data Extractor
3// @description Extract extension data from Discord posts or GitHub repos and copy as formatted JSON (Alt+Shift+E)
4// @version 1.1.2
5// @match https://*.discord.com/*
6// @match https://github.com/*/*
7// @icon 
8// ==/UserScript==
9(function() {
10 'use strict';
11
12 // Utility function to show notifications
13 function showNotification(message, isError = false) {
14 const notification = document.createElement('div');
15 notification.textContent = message;
16 notification.style.cssText = `
17 position: fixed;
18 top: 20px;
19 right: 20px;
20 background: ${isError ? '#dc3545' : '#28a745'};
21 color: white;
22 padding: 15px 20px;
23 border-radius: 8px;
24 box-shadow: 0 4px 12px rgba(0,0,0,0.3);
25 z-index: 999999;
26 font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, sans-serif;
27 font-size: 14px;
28 max-width: 400px;
29 word-wrap: break-word;
30 animation: slideIn 0.3s ease-out;
31 `;
32
33 const style = document.createElement('style');
34 style.textContent = `
35 @keyframes slideIn {
36 from { transform: translateX(400px); opacity: 0; }
37 to { transform: translateX(0); opacity: 1; }
38 }
39 `;
40 document.head.appendChild(style);
41
42 document.body.appendChild(notification);
43
44 setTimeout(() => {
45 notification.style.transition = 'opacity 0.3s ease-out';
46 notification.style.opacity = '0';
47 setTimeout(() => notification.remove(), 300);
48 }, 4000);
49 }
50
51 // Check if we're on a supported page
52 function isSupportedPage() {
53 const url = window.location.href;
54 const isDiscord = /https:\/\/discord\.com\/channels\/1100685673633153084\/\d+/.test(url);
55 const isGitHub = /https:\/\/github\.com\/[^\/]+\/[^\/]+/.test(url);
56 return isDiscord || isGitHub;
57 }
58
59 // Extract GitHub URL from Discord post
60 function extractGitHubUrl(postContent) {
61 const urlPattern = /https:\/\/github\.com\/[^\/\s\)]+\/[^\/\s\)]+\/?/g;
62 const matches = postContent.match(urlPattern);
63 if (matches && matches.length > 0) {
64 return matches[0].replace(/\/$/, ''); // Remove trailing slash
65 }
66 return null;
67 }
68
69 // Extract ID from GitHub URL
70 function extractIdFromUrl(url) {
71 const match = url.match(/github\.com\/[^\/]+\/([^\/\s]+)/);
72 if (match) {
73 return match[1].replace(/\/$/, '');
74 }
75 return null;
76 }
77
78 // Clean extension name
79 function cleanExtensionName(id) {
80 let name = id;
81 // Remove common prefixes/suffixes
82 name = name.replace(/^(SillyTavern-|ST-)/i, '');
83 name = name.replace(/(-Extension|-Ext|Extension|Ext)$/i, '');
84 // Add spaces before capital letters
85 name = name.replace(/([a-z])([A-Z])/g, '$1 $2');
86 // Replace hyphens and underscores with spaces
87 name = name.replace(/[-_]/g, ' ');
88 // Title case
89 name = name.split(' ').map(word =>
90 word.charAt(0).toUpperCase() + word.slice(1).toLowerCase()
91 ).join(' ');
92 return name.trim();
93 }
94
95 // Extract Discord post content
96 async function extractDiscordPostContent() {
97 try {
98 console.log('Extracting Discord post content...');
99
100 // Wait for content to load
101 await new Promise(resolve => setTimeout(resolve, 1000));
102
103 // Try multiple selectors to find the initial post
104 const selectors = [
105 'article[id^="chat-messages-"]',
106 'li[id^="chat-messages-"]',
107 'div[class*="message_"]',
108 'div[class*="cozyMessage"]'
109 ];
110
111 let postElement = null;
112 for (const selector of selectors) {
113 const elements = document.querySelectorAll(selector);
114 if (elements.length > 0) {
115 postElement = elements[0];
116 break;
117 }
118 }
119
120 if (!postElement) {
121 throw new Error('Could not find Discord post element');
122 }
123
124 const postContent = postElement.textContent || postElement.innerText;
125 console.log('Extracted post content:', postContent.substring(0, 200));
126
127 const githubUrl = extractGitHubUrl(postContent);
128 if (!githubUrl) {
129 throw new Error('No GitHub URL found in post');
130 }
131
132 console.log('Found GitHub URL:', githubUrl);
133
134 const id = extractIdFromUrl(githubUrl);
135 if (!id) {
136 throw new Error('Could not extract ID from GitHub URL');
137 }
138
139 // Use AI to generate name and description
140 const namePrompt = `Convert this repository ID to a clean extension name: "${id}". Remove prefixes like "SillyTavern-", "ST-" and suffixes like "-Extension", "Extension", "Ext". Add spaces and use title case. Return ONLY the clean name, nothing else.`;
141
142 const descriptionPrompt = `Summarize this SillyTavern extension post in 1-2 concise sentences. Use plain text only - NO markdown formatting, NO asterisks, NO special characters, NO newlines. If there are warnings about WIP status, dependencies, requirements, or compatibility issues, prepend them as "Warning: [concise warning]. " Here is the post content:\n\n${postContent.substring(0, 2000)}`;
143
144 console.log('Calling AI for name and description...');
145 const [name, description] = await Promise.all([
146 RM.aiCall(namePrompt),
147 RM.aiCall(descriptionPrompt)
148 ]);
149
150 console.log('AI generated name:', name);
151 console.log('AI generated description:', description);
152
153 return {
154 id,
155 type: 'extension',
156 name: name.trim(),
157 description: description.trim().replace(/\n/g, ' ').replace(/\s+/g, ' '),
158 url: githubUrl
159 };
160
161 } catch (error) {
162 console.error('Error extracting Discord post:', error);
163 throw error;
164 }
165 }
166
167 // Extract GitHub repository data
168 async function extractGitHubRepoData() {
169 try {
170 console.log('Extracting GitHub repository data...');
171
172 const url = window.location.href.replace(/\/$/, '');
173 const id = extractIdFromUrl(url);
174
175 if (!id) {
176 throw new Error('Could not extract repository ID from URL');
177 }
178
179 // Try to get About section
180 let description = '';
181 const aboutSelectors = [
182 'p[class*="About"]',
183 'p.f4.my-3',
184 '[data-pjax="#repo-content-pjax-container"] p'
185 ];
186
187 let aboutElement = null;
188 for (const selector of aboutSelectors) {
189 aboutElement = document.querySelector(selector);
190 if (aboutElement && aboutElement.textContent.trim()) {
191 description = aboutElement.textContent.trim();
192 console.log('Found About section:', description);
193 break;
194 }
195 }
196
197 // If no About section, try to get README
198 if (!description) {
199 console.log('No About section found, looking for README...');
200 const readmeSelectors = [
201 'article[class*="markdown"]',
202 'div[class*="markdown-body"]',
203 '#readme'
204 ];
205
206 let readmeElement = null;
207 for (const selector of readmeSelectors) {
208 readmeElement = document.querySelector(selector);
209 if (readmeElement) {
210 const readmeText = readmeElement.textContent || readmeElement.innerText;
211 console.log('Found README, length:', readmeText.length);
212
213 // Use AI to summarize README
214 const descriptionPrompt = `Summarize this SillyTavern extension README in 1-2 concise sentences. Use plain text only - NO markdown formatting, NO asterisks, NO special characters, NO newlines. If there are warnings about WIP status, dependencies, requirements, or compatibility issues, prepend them as "Warning: [concise warning]. " Here is the README:\n\n${readmeText.substring(0, 2000)}`;
215
216 description = await RM.aiCall(descriptionPrompt);
217 console.log('AI generated description from README:', description);
218 break;
219 }
220 }
221 }
222
223 if (!description) {
224 throw new Error('Could not find repository description or README');
225 }
226
227 // Generate clean name
228 const namePrompt = `Convert this repository ID to a clean extension name: "${id}". Remove prefixes like "SillyTavern-", "ST-" and suffixes like "-Extension", "Extension", "Ext". Add spaces and use title case. Return ONLY the clean name, nothing else.`;
229 const name = await RM.aiCall(namePrompt);
230
231 console.log('AI generated name:', name);
232
233 return {
234 id,
235 type: 'extension',
236 name: name.trim(),
237 description: description.trim().replace(/\n/g, ' ').replace(/\s+/g, ' '),
238 url
239 };
240
241 } catch (error) {
242 console.error('Error extracting GitHub repo:', error);
243 throw error;
244 }
245 }
246
247 // Format data as JSON
248 function formatAsJson(data) {
249 const json = {
250 id: data.id,
251 type: data.type,
252 name: data.name,
253 description: data.description,
254 url: data.url
255 };
256
257 // Manually format with 8 spaces for fields and trailing comma
258 const lines = [
259 '{',
260 ` "id": "${json.id}",`,
261 ` "type": "${json.type}",`,
262 ` "name": "${json.name}",`,
263 ` "description": "${json.description}",`,
264 ` "url": "${json.url}"`,
265 '},'
266 ];
267
268 return lines.join('\n');
269 }
270
271 // Copy to clipboard
272 async function copyToClipboard(text) {
273 try {
274 if (navigator.clipboard && navigator.clipboard.writeText) {
275 await navigator.clipboard.writeText(text);
276 } else {
277 // Fallback for browsers without clipboard API
278 const textarea = document.createElement('textarea');
279 textarea.value = text;
280 textarea.style.position = 'fixed';
281 textarea.style.opacity = '0';
282 document.body.appendChild(textarea);
283 textarea.select();
284 document.execCommand('copy');
285 document.body.removeChild(textarea);
286 }
287 } catch (error) {
288 console.error('Clipboard error:', error);
289 throw new Error('Failed to copy to clipboard. Please check browser permissions.');
290 }
291 }
292
293 // Main extraction function
294 async function extractAndCopy() {
295 try {
296 console.log('Starting extraction...');
297
298 // Show processing notification
299 showNotification('Processing... Please wait while AI extracts the data.');
300
301 const url = window.location.href;
302 let data;
303
304 if (/https:\/\/discord\.com\/channels\/1100685673633153084/.test(url)) {
305 console.log('Detected Discord page');
306 data = await extractDiscordPostContent();
307 } else if (/https:\/\/github\.com\/[^\/]+\/[^\/]+/.test(url)) {
308 console.log('Detected GitHub page');
309 data = await extractGitHubRepoData();
310 } else {
311 throw new Error('Unsupported page. This extension only works on Discord posts and GitHub repositories.');
312 }
313
314 const formattedJson = formatAsJson(data);
315 console.log('Formatted JSON:', formattedJson);
316
317 await copyToClipboard(formattedJson);
318
319 showNotification(`Data about ${data.id} successfully JSON formatted and copied to clipboard`);
320
321 } catch (error) {
322 console.error('Extraction error:', error);
323 showNotification(`Error: ${error.message}`, true);
324 }
325 }
326
327 // Initialize
328 function init() {
329 console.log('SillyTavern Extension Data Extractor loaded');
330
331 if (!isSupportedPage()) {
332 console.log('Not on a supported page');
333 return;
334 }
335
336 // Show activation notification
337 setTimeout(() => {
338 showNotification('Extension active! Press Alt+Shift+E to extract data');
339 }, 1000);
340
341 // Add keyboard shortcut listener
342 document.addEventListener('keydown', (event) => {
343 // Alt+Shift+E
344 if (event.altKey && event.shiftKey && event.key.toLowerCase() === 'e') {
345 event.preventDefault();
346 console.log('Keyboard shortcut triggered');
347 extractAndCopy();
348 }
349 });
350
351 console.log('Keyboard shortcut registered: Alt+Shift+E');
352 }
353
354 // Wait for page to load
355 if (document.readyState === 'loading') {
356 document.addEventListener('DOMContentLoaded', init);
357 } else {
358 init();
359 }
360
361})();