SillyTavern Extension Data Extractor

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})();
SillyTavern Extension Data Extractor | Robomonkey