Last active
January 26, 2025 13:45
-
-
Save san-kumar/b3604ca40905556b4135fee64ead58f2 to your computer and use it in GitHub Desktop.
tampermonkey script to show summaries on hacker news
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
// ==UserScript== | |
// @name hn summary (kagi) | |
// @namespace http://tampermonkey.net/ | |
// @version 0.1 | |
// @description try to take over the world! | |
// @author You | |
// @match https://news.ycombinator.com/* | |
// @icon https://www.google.com/s2/favicons?sz=64&domain=ycombinator.com | |
// @grant GM_xmlhttpRequest | |
// @require https://code.jquery.com/jquery-3.4.1.min.js | |
// ==/UserScript== | |
(async function () { | |
'use strict'; | |
let getSummary = async function(url, e, type, fails = 0) { | |
let loading = $(`<img class="load_img" src="" height="12" />`); | |
e.after(loading); | |
let result; | |
if(type == 'summary') { | |
if(!fails) { | |
let queue = await new Promise(r => GM_xmlhttpRequest ( { | |
method: "POST", | |
url: "https://labs.kagi.com/v1/summarization", | |
data: JSON.stringify({url}), | |
headers: { "accept": "application/json", "content-type": "application/json"}, | |
onload: function (response) { | |
let obj = JSON.parse(response.responseText); | |
console.log(obj); | |
r(); | |
} | |
})); | |
} | |
result = await new Promise(r => GM_xmlhttpRequest ( { | |
method: "GET", | |
url: "https://labs.kagi.com/v1/summary_status?url=" + encodeURIComponent(url), | |
onload: function (response) { | |
let obj = JSON.parse(response.responseText); | |
console.log(obj); | |
if (obj.summary) { | |
r(obj.summary); | |
} else if(fails < 3) { | |
console.log("retry: ", fails); | |
setTimeout(() => getSummary(url, e, true, fails+1), 10000); | |
} | |
} | |
})); | |
} else { | |
console.log({url}); | |
result = await new Promise(r => GM_xmlhttpRequest ( { | |
method: "POST", | |
url: "https://labs.kagi.com/v1/takeaways", | |
data: JSON.stringify({url}), | |
headers: { "accept": "application/json", "content-type": "application/json"}, | |
onload: function (response) { | |
let obj = JSON.parse(response.responseText); | |
console.log(obj); | |
r(obj.takeaways); | |
} | |
})); | |
} | |
let close = $(`<a href="#" onclick="return false;" style="position: absolute; top: 0; right: 5px;">✖</a>`); | |
let takeaway = $(`<tr><td colspan=2></td><td style="padding: 10px 0; color: #333;"><div style="border: 1px solid silver;max-width:min(90vw, 640px); padding: 10px;border-radius: 5px; position: relative;">${result}</div></td></tr>`); | |
e.closest('tr').after(takeaway); | |
takeaway.find('div').append(close); | |
takeaway.find('a').click(() => takeaway.remove()); | |
$('.load_img').remove(); | |
}; | |
$('a:contains(flag)').each((i, e) => { | |
let flag = $(e); | |
let url = flag.closest('tr').prev().find('.title a[href]').first().attr('href'); | |
let comments = flag.closest('tr').find('a:contains(comments)').first().get(0)?.href; | |
if(!url) { | |
url = location.href; | |
} | |
let summary = $(`<a href="https://labs.kagi.com/ai/sum?url=${url}&expand=1" onclick="return false">summary</a>`); | |
summary.click(() => getSummary(url, summary, 'summary')); | |
flag.before(summary); | |
flag.before(" | "); | |
let takeaways = $(`<a href="https://labs.kagi.com/ai/sum?url=${url}&expand=1" onclick="return false">takeaways</a>`); | |
takeaways.click(() => getSummary(url, summary, 'takeaways')); | |
flag.before(takeaways); | |
flag.before(" | "); | |
if(comments) { | |
let hn = $(`<a href="https://labs.kagi.com/ai/sum?url=${url}&expand=1" onclick="return false">hn</a>`); | |
hn.click(() => getSummary(comments, summary, 'takeaways')); | |
flag.before(hn); | |
flag.before(" | "); | |
} | |
let last; | |
flag.closest('tr').find('a:contains(flag)').remove(); | |
flag.closest('tr').find('a:contains(hide)').remove(); | |
flag.closest('tr').find('a:contains(hours ago)').remove(); | |
flag.closest('.subline').contents().filter(function() { | |
let prev = last === true; | |
last = (this.nodeType === 3) && ($(this).text().trim() === '|'); | |
return prev && last; | |
}).remove(); | |
}); | |
})(); |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment