Add Markdown conversion option
This commit is contained in:
		
					parent
					
						
							
								5e0356086a
							
						
					
				
			
			
				commit
				
					
						200c03c875
					
				
			
		
					 6 changed files with 37 additions and 3 deletions
				
			
		|  | @ -14,6 +14,7 @@ message meets a specified criterion. | ||||||
| - **Custom system prompts** – tailor the instructions sent to the model for more precise results. | - **Custom system prompts** – tailor the instructions sent to the model for more precise results. | ||||||
| - **Persistent result caching** – classification results and reasoning are saved to disk so messages aren't re-evaluated across restarts. | - **Persistent result caching** – classification results and reasoning are saved to disk so messages aren't re-evaluated across restarts. | ||||||
| - **Advanced parameters** – tune generation settings like temperature, top‑p and more from the options page. | - **Advanced parameters** – tune generation settings like temperature, top‑p and more from the options page. | ||||||
|  | - **Markdown conversion** – optionally convert HTML bodies to Markdown before sending them to the AI service. | ||||||
| - **Debug logging** – optional colorized logs help troubleshoot interactions with the AI service. | - **Debug logging** – optional colorized logs help troubleshoot interactions with the AI service. | ||||||
| - **Automatic rules** – create rules that tag or move new messages based on AI classification. | - **Automatic rules** – create rules that tag or move new messages based on AI classification. | ||||||
| - **Rule ordering** – drag rules to prioritize them and optionally stop processing after a match. | - **Rule ordering** – drag rules to prioritize them and optionally stop processing after a match. | ||||||
|  |  | ||||||
|  | @ -15,4 +15,5 @@ | ||||||
|   "template.custom": { "message": "Custom" }, |   "template.custom": { "message": "Custom" }, | ||||||
|   "options.save": { "message": "Save" }, |   "options.save": { "message": "Save" }, | ||||||
|   "options.debugLogging": { "message": "Enable debug logging" } |   "options.debugLogging": { "message": "Enable debug logging" } | ||||||
|  |   ,"options.htmlToMarkdown": { "message": "Convert HTML body to Markdown" } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -22,6 +22,8 @@ let iconTimer = null; | ||||||
| let timingStats = { count: 0, mean: 0, m2: 0, total: 0, last: -1 }; | let timingStats = { count: 0, mean: 0, m2: 0, total: 0, last: -1 }; | ||||||
| let currentStart = 0; | let currentStart = 0; | ||||||
| let logGetTiming = true; | let logGetTiming = true; | ||||||
|  | let htmlToMarkdown = false; | ||||||
|  | let TurndownService = null; | ||||||
| 
 | 
 | ||||||
| function setIcon(path) { | function setIcon(path) { | ||||||
|     if (browser.browserAction) { |     if (browser.browserAction) { | ||||||
|  | @ -70,7 +72,17 @@ function collectText(part, bodyParts, attachments) { | ||||||
|         attachments.push(`${name} (${ct}, ${part.size || byteSize(body)} bytes)`); |         attachments.push(`${name} (${ct}, ${part.size || byteSize(body)} bytes)`); | ||||||
|     } else if (ct.startsWith("text/html")) { |     } else if (ct.startsWith("text/html")) { | ||||||
|         const doc = new DOMParser().parseFromString(body, 'text/html'); |         const doc = new DOMParser().parseFromString(body, 'text/html'); | ||||||
|         bodyParts.push(replaceInlineBase64(doc.body.textContent || "")); |         if (htmlToMarkdown && TurndownService) { | ||||||
|  |             try { | ||||||
|  |                 const td = new TurndownService(); | ||||||
|  |                 const md = td.turndown(doc.body.innerHTML || body); | ||||||
|  |                 bodyParts.push(replaceInlineBase64(`[HTML Body converted to Markdown]\n${md}`)); | ||||||
|  |             } catch (e) { | ||||||
|  |                 bodyParts.push(replaceInlineBase64(doc.body.textContent || "")); | ||||||
|  |             } | ||||||
|  |         } else { | ||||||
|  |             bodyParts.push(replaceInlineBase64(doc.body.textContent || "")); | ||||||
|  |         } | ||||||
|     } else { |     } else { | ||||||
|         bodyParts.push(replaceInlineBase64(body)); |         bodyParts.push(replaceInlineBase64(body)); | ||||||
|     } |     } | ||||||
|  | @ -213,16 +225,19 @@ async function clearCacheForMessages(idsInput) { | ||||||
|     try { |     try { | ||||||
|         AiClassifier = await import(browser.runtime.getURL("modules/AiClassifier.js")); |         AiClassifier = await import(browser.runtime.getURL("modules/AiClassifier.js")); | ||||||
|         logger.aiLog("AiClassifier imported", {debug: true}); |         logger.aiLog("AiClassifier imported", {debug: true}); | ||||||
|  |         const td = await import(browser.runtime.getURL("resources/js/turndown.js")); | ||||||
|  |         TurndownService = td.default || td.TurndownService; | ||||||
|     } catch (e) { |     } catch (e) { | ||||||
|         console.error("failed to import AiClassifier", e); |         console.error("failed to import AiClassifier", e); | ||||||
|         return; |         return; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     try { |     try { | ||||||
|         const store = await storage.local.get(["endpoint", "templateName", "customTemplate", "customSystemPrompt", "aiParams", "debugLogging", "aiRules"]); |         const store = await storage.local.get(["endpoint", "templateName", "customTemplate", "customSystemPrompt", "aiParams", "debugLogging", "htmlToMarkdown", "aiRules"]); | ||||||
|         logger.setDebug(store.debugLogging); |         logger.setDebug(store.debugLogging); | ||||||
|         await AiClassifier.setConfig(store); |         await AiClassifier.setConfig(store); | ||||||
|         await AiClassifier.init(); |         await AiClassifier.init(); | ||||||
|  |         htmlToMarkdown = store.htmlToMarkdown === true; | ||||||
|         const savedStats = await storage.local.get('classifyStats'); |         const savedStats = await storage.local.get('classifyStats'); | ||||||
|         if (savedStats.classifyStats && typeof savedStats.classifyStats === 'object') { |         if (savedStats.classifyStats && typeof savedStats.classifyStats === 'object') { | ||||||
|             Object.assign(timingStats, savedStats.classifyStats); |             Object.assign(timingStats, savedStats.classifyStats); | ||||||
|  | @ -254,6 +269,10 @@ async function clearCacheForMessages(idsInput) { | ||||||
|                 }); |                 }); | ||||||
|                 logger.aiLog("aiRules updated from storage change", {debug: true}, aiRules); |                 logger.aiLog("aiRules updated from storage change", {debug: true}, aiRules); | ||||||
|             } |             } | ||||||
|  |             if (changes.htmlToMarkdown) { | ||||||
|  |                 htmlToMarkdown = changes.htmlToMarkdown.newValue === true; | ||||||
|  |                 logger.aiLog("htmlToMarkdown updated from storage change", {debug: true}, htmlToMarkdown); | ||||||
|  |             } | ||||||
|         }); |         }); | ||||||
|     } catch (err) { |     } catch (err) { | ||||||
|         logger.aiLog("failed to load config", {level: 'error'}, err); |         logger.aiLog("failed to load config", {level: 'error'}, err); | ||||||
|  |  | ||||||
|  | @ -98,6 +98,11 @@ | ||||||
|                             <input type="checkbox" id="debug-logging"> Enable debug logging |                             <input type="checkbox" id="debug-logging"> Enable debug logging | ||||||
|                         </label> |                         </label> | ||||||
|                     </div> |                     </div> | ||||||
|  |                     <div class="field"> | ||||||
|  |                         <label class="checkbox"> | ||||||
|  |                             <input type="checkbox" id="html-to-markdown"> Convert HTML body to Markdown | ||||||
|  |                         </label> | ||||||
|  |                     </div> | ||||||
|                     <div class="field"> |                     <div class="field"> | ||||||
|                         <label class="label" for="max_tokens">Max tokens</label> |                         <label class="label" for="max_tokens">Max tokens</label> | ||||||
|                         <div class="control"> |                         <div class="control"> | ||||||
|  |  | ||||||
|  | @ -9,6 +9,7 @@ document.addEventListener('DOMContentLoaded', async () => { | ||||||
|         'customSystemPrompt', |         'customSystemPrompt', | ||||||
|         'aiParams', |         'aiParams', | ||||||
|         'debugLogging', |         'debugLogging', | ||||||
|  |         'htmlToMarkdown', | ||||||
|         'aiRules', |         'aiRules', | ||||||
|         'aiCache' |         'aiCache' | ||||||
|     ]); |     ]); | ||||||
|  | @ -81,6 +82,9 @@ document.addEventListener('DOMContentLoaded', async () => { | ||||||
|     const debugToggle = document.getElementById('debug-logging'); |     const debugToggle = document.getElementById('debug-logging'); | ||||||
|     debugToggle.checked = defaults.debugLogging === true; |     debugToggle.checked = defaults.debugLogging === true; | ||||||
| 
 | 
 | ||||||
|  |     const htmlToggle = document.getElementById('html-to-markdown'); | ||||||
|  |     htmlToggle.checked = defaults.htmlToMarkdown === true; | ||||||
|  | 
 | ||||||
|     const aiParams = Object.assign({}, DEFAULT_AI_PARAMS, defaults.aiParams || {}); |     const aiParams = Object.assign({}, DEFAULT_AI_PARAMS, defaults.aiParams || {}); | ||||||
|     for (const [key, val] of Object.entries(aiParams)) { |     for (const [key, val] of Object.entries(aiParams)) { | ||||||
|         const el = document.getElementById(key); |         const el = document.getElementById(key); | ||||||
|  | @ -395,6 +399,7 @@ document.addEventListener('DOMContentLoaded', async () => { | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|         const debugLogging = debugToggle.checked; |         const debugLogging = debugToggle.checked; | ||||||
|  |         const htmlToMarkdown = htmlToggle.checked; | ||||||
|         const rules = [...rulesContainer.querySelectorAll('.rule')].map(ruleEl => { |         const rules = [...rulesContainer.querySelectorAll('.rule')].map(ruleEl => { | ||||||
|             const criterion = ruleEl.querySelector('.criterion').value; |             const criterion = ruleEl.querySelector('.criterion').value; | ||||||
|             const actions = [...ruleEl.querySelectorAll('.action-row')].map(row => { |             const actions = [...ruleEl.querySelectorAll('.action-row')].map(row => { | ||||||
|  | @ -413,7 +418,7 @@ document.addEventListener('DOMContentLoaded', async () => { | ||||||
|             const stopProcessing = ruleEl.querySelector('.stop-processing')?.checked; |             const stopProcessing = ruleEl.querySelector('.stop-processing')?.checked; | ||||||
|             return { criterion, actions, stopProcessing }; |             return { criterion, actions, stopProcessing }; | ||||||
|         }).filter(r => r.criterion); |         }).filter(r => r.criterion); | ||||||
|         await storage.local.set({ endpoint, templateName, customTemplate: customTemplateText, customSystemPrompt, aiParams: aiParamsSave, debugLogging, aiRules: rules }); |         await storage.local.set({ endpoint, templateName, customTemplate: customTemplateText, customSystemPrompt, aiParams: aiParamsSave, debugLogging, htmlToMarkdown, aiRules: rules }); | ||||||
|         try { |         try { | ||||||
|             await AiClassifier.setConfig({ endpoint, templateName, customTemplate: customTemplateText, customSystemPrompt, aiParams: aiParamsSave, debugLogging }); |             await AiClassifier.setConfig({ endpoint, templateName, customTemplate: customTemplateText, customSystemPrompt, aiParams: aiParamsSave, debugLogging }); | ||||||
|             logger.setDebug(debugLogging); |             logger.setDebug(debugLogging); | ||||||
|  |  | ||||||
|  | @ -972,3 +972,6 @@ var TurndownService = (function () { | ||||||
|   return TurndownService; |   return TurndownService; | ||||||
| 
 | 
 | ||||||
| }()); | }()); | ||||||
|  | 
 | ||||||
|  | export { TurndownService }; | ||||||
|  | export default TurndownService; | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue