Skip to content

Commit

Permalink
Adding "publish-to-gts" script (#245)
Browse files Browse the repository at this point in the history
* Added Ascii Tree Factory

* Update readme.md

the `l` got lost in the `.qml` extension...

* Added ToDo section to readme file

* Update readme.md

* Update info.json

Build was failing for inconsistent identifier

* Update info.json

* Update and rename ASCII-Tree-Factory.qml to ascii-tree-factory.qml

* Update info.json

Removed Linux and MacOs as not tested

* Update ascii-tree-factory.qml

regression bug due to changing identifier

* Update info.json

* fix: missing else

* Fix: same levels now connect better
Enhancement: style selector
Enhancement: separator selector

	modified:   ascii-tree-factory/ASCII-Tree-Factory.qml
	modified:   ascii-tree-factory/info.json
	modified:   ascii-tree-factory/readme.md

* Fix: filename and indetifier casing
	renamed:    ascii-tree-factory/ASCII-Tree-Factory.qml -> ascii-tree-factory/ascii-tree-factory.qml
	modified:   ascii-tree-factory/readme.md

* info.json: Added extended description

* Fix: description adjustment, it was too long

* Update info.jsonversion update

* Added from Codeberg personal repo

* Fix: minor updates to names, versions; Add "string-secret" for Authorization Code script setting.

---------

Co-authored-by: 77nn <[email protected]>
  • Loading branch information
77nnit and 77nn authored Nov 4, 2024
1 parent 2a2b721 commit 819bbd0
Show file tree
Hide file tree
Showing 3 changed files with 474 additions and 0 deletions.
10 changes: 10 additions & 0 deletions publish-to-gts/info.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,10 @@
{
"name": "Publish to GoToSocial",
"identifier": "publish-to-gts",
"script": "publish-to-gts.qml",
"authors": ["@77nnit"],
"platforms": ["linux", "windows"],
"version": "0.1.1",
"minAppVersion": "24.11.0",
"description" : "Publish To GtS lets you publish your notes to your <a hfref='https://gotosocial.org'>GoToSocial</a> account."
}
379 changes: 379 additions & 0 deletions publish-to-gts/publish-to-gts.qml
Original file line number Diff line number Diff line change
@@ -0,0 +1,379 @@
import QtQml 2.0
import QOwnNotesTypes 1.0

Script {
property string serverInstance;
property string authCode;
property string visibility;
property bool local_only;
property bool sensitive;
property string spoiler_text;
property string language;

property variant settingsVariables: [
{
"identifier": "serverInstance",
"name": "Server instance",
"description": "Server instance you want to connect to (i.e.: example.org - no spaces, no protocol, no slashes)",
"type": "string",
"default": "example.org",
},
{
"identifier": "authCode",
"name": "Authentication Code",
"description": "Code returned by GtS after performing a successful authentication, if you paste it here you won't need to authenticate again until expiry",
"type": "string-secret",
"default": "",
},
{
"identifier": "visibility",
"name": "Visibility",
"description": "Default visibility for published posts",
"type": "selection",
"default": "public",
"items": {"public": "Public", "unlisted": "Unlisted", "private": "Private", "mutuals_only": "Mutuals", "direct": "Direct Message"},
},
{
"identifier": "local_only",
"name": "Local only",
"description": "If the post is local only, it will not be seen from federated instances",
"text": "Yes, let it be Local Only",
"type": "boolean",
"default": false,
},
{
"identifier": "sensitive",
"name": "Content Warning",
"description": "The post text will not be immediately visible, as it may be sensible to some audience",
"text": "Yes, the post content is usually sensible",
"type": "boolean",
"default": false,
},
{
"identifier": "spoiler_text",
"name": "Content Warning text",
"description": "Text to show as a content warning for senstitive posts",
"type": "string",
"default": "Sensible content ahead!",
},
{
"identifier": "language",
"name": "Language code",
"description": "2-chars laguage code as per <a href='https://www.loc.gov/standards/iso639-2/php/English_list.php'>https://www.loc.gov/standards/iso639-2/php/English_list.php</a>:",
"type": "string",
"default": "en",
},
];

property variant currentParams: {
"visibility": null,
"local_only": null,
"sensitive": null,
"spoiler_text": null,
"language": null
}
property variant additionalParams: {
"created_at": null,
"id": null,
"url": null
}

function init() {
script.registerCustomAction("publish", "Publish current note to GtS","",true,true,false);
script.registerCustomAction("newPost", "New post for GtS","",true,true,false);

//validate server instance
if (!(script.getPersistentVariable("publishToGts/"+serverInstance))){
serverInstance = serverInstance.match(/(?!(\w+:\/\/))(\w+.)*(\w+)/g)[0];
}
}

// This function returns a true or a false or a string that doesn't match true or false
function parseBool(val) {
if (Object.prototype.toString.call(val)=="[object String]"){
return val == "true" ? true : (val == "false" ? false : val);
} else {
return val === true || val === "true";
}
}

// This function generates a Post Header with comments and default parameter values for the current note.
// if includeAdditional = true adds additional attributes (like created_at for already published notes)
function generatePostHeader(includeAdditional){
let postHeader = "";
postHeader += "***Publish to GtS - Post Header***\n";
postHeader += "\n";
postHeader += "#Edit the values to adjust the post settings.\n";
postHeader += "#Missing properties will be defaulted as per script settings.\n";
postHeader += "#Confirmation will be asked before publishing.\n";
postHeader += "#This section will not be published.\n";
postHeader += "\n";
// checking all post parameters in currentParams against user settings params
Object.keys(currentParams).forEach(function(key){
// using .every to break out the cycle
settingsVariables.every(function(varObj){
if (key == varObj.identifier){
// the eval used here is safe, as the variable it evaluates contains always a string value.
postHeader += `${key}: ${eval(varObj.identifier)}${"\n"}`;
return false;
} else {
return true;
}
});
});
if (includeAdditional){
Object.keys(additionalParams).forEach(function(key){
postHeader += `${key}: ${additionalParams[key]}${"\n"}`;
});
}
return postHeader;
}

function updatePostHeader(){
let current = script.currentNote()
let sections = current.noteText.split("---");
script.tagCurrentNote("Pub2GtS");
if (sections && sections[1]){
script.triggerMenuAction("actionAllow_note_editing", 1);
mainWindow.focusNoteTextEdit();
script.noteTextEditSetCursorPosition(0);
script.noteTextEditSelectAll();
script.noteTextEditWrite([sections[0], ("---\n" + generatePostHeader(true) + "\n---")].concat(sections.slice(2)).join(""));
}
}

// function that reads a Post Header section and populate the currentParams object
function decodePostHeader(){
let current = script.currentNote(); // current note
let postParams = {};
let postHeader = current.noteText.split("---");
if (postHeader[1]){
postHeader[1].split("\n").forEach(function(param){
if (! param.startsWith("#")){
let thisParam = param.split(":");
if (thisParam[0] && thisParam[1]){
postParams[thisParam[0].trim()]=param.split(":").slice(1).join("").trim();
}
}
});
currentParams = postParams;
return true;
} else {
script.log("Publish to GtS: Current note does not have a valid Post Header.");
return false;
}
}

// This function shows a confirmation message box with current note posting parameters
function confirmPublish(){
let msgText = "<b>You are about to publish the current note</b> with the following settings:\n<ul>";
Object.keys(currentParams).forEach(function(key){
msgText += "<li>" + key + ": " + currentParams[key] + "</li>";
});
msgText += "</ul>";
msgText += `Press "<b>Ok</b>" to confirm and post the note with the above settings;`
msgText += `Press "<b>Cancel</b>" to continue editing your note.`;
return (script.questionMessageBox(msgText, "Publish to GtS: confirm action", 0x00000400|0x00400000) == 1024);
}

// This function ask user confirmation to generate a postHeader for the current post
function confirmPostHeader(){
let msgText = `The current note does not appear to have a valid Post Header for publishing. You can:<ul>`;
msgText += `<li>Press "<b>OK</b>" to publish the note with default settings</li>`;
msgText += `<li>Press "<b>Apply</b>" to cancel the publishing and generate a Post Header for your note</li>`;
msgText += `<li>Press "<b>Cancel</b>" to cancel the publishing and add a Post Header manually</li></ul>`;
msgText += `<b>Note:</b>Generating a Post Header will add the header <i>above</i> the title, if present.`;
return script.questionMessageBox(msgText, "Publish to GtS: missing Post Header", 0x00000400|0x02000000|0x00400000);
}

function confirmDuplicatePost(){
let msgText = `The current note Post Header contains "created_at" property. This may indicate that the note was already published. Are you sure you want to publish the note <b>again</b>?`;
return (script.questionMessageBox(msgText, "Publish to GtS: publish duplicate note", 0x00000400|0x00400000)==1024);
}

// This function performs all API endpoint requests and returns text responses
function request(verb, endpoint, aT, data){
// create request
let xhr = new XMLHttpRequest();
let url = "https://" + serverInstance + endpoint;
// open synchronous request
xhr.open(verb, url, false);
// setting content type request header
xhr.setRequestHeader("Content-Type", "application/json; charset=utf-8");
// is accessToken is provided in the function call adding the Authorization header
if (aT && aT.length > 0){
xhr.setRequestHeader("Authorization", "Bearer " + aT);
}
// sending the request with data, if date is there
if (data){
// xhr.setRequestHeader("Content-Length", JSON.stringify(data).length);
xhr.send(JSON.stringify(data));
} else {
xhr.send();
}
// since tha call is synchronous at this poit we already have the response
// the response body is returned
if (xhr.status == 200){
return xhr.response;
// If the request is nor completed with a success code 200, write to che console the error and return null
} else {
script.log ("Publish to GtS: Error: " + serverInstance + endpoint + " returned code " + xhr.status + " - " + xhr.statusText);
script.log (JSON.stringify(xhr.response));
script.log (xhr.getAllResponseHeaders());
return null;
}
}

// This function wraps the "verify credential" call
function verifyCredentials(aT){
// calling the request over the specific endpoint, passing the accessToken
let res = request ("GET", "/api/v1/accounts/verify_credentials", aT);
return res && true;
}

function customActionInvoked(identifier) {

//handler for newPost command
if (identifier == "newPost"){
let date = new Date();
let headline = "Note " + date.toISOString();
script.createNote("# " + headline + "\n\n---\n\n" + generatePostHeader() + "\n\n---\n\n");
let currentNote = script.currentNote();
script.triggerMenuAction("actionAllow_note_editing", 1);
currentNote.renameNoteFile(headline);
mainWindow.focusNoteTextEdit();
script.tagCurrentNote("Pub2GtS");
return;
}

// handler for publish command
if (identifier == "publish") {
// local variables init
let clientName = "QONPublishToGts";
let clientMode = "Read+Write";
let clientId = "";
let clientSecret = "";
let credentialsVerified = false;
let current = script.currentNote(); // current note
let currentPost = ""; //current part of note that represents a post when post header stripped

if (!decodePostHeader()){
let exitCondition = true;
let confirmResult = confirmPostHeader();
switch (confirmResult){
case 1024:{
script.log("Publish to GtS: default posting settings confirmed.");
exitCondition = false;
break;
}
case 33554432:{
script.log("Publish to GtS: generating default Post Header.");
// attach Post Header at the beginning of post
script.triggerMenuAction("actionAllow_note_editing", 1);
mainWindow.focusNoteTextEdit();
script.noteTextEditSetCursorPosition(0);
let date = new Date();
let headline = "Note " + date.toISOString();
script.noteTextEditWrite("# " + headline + "\n\n---\n\n" + generatePostHeader() + "\n\n---\n\n");
script.tagCurrentNote("Pub2GtS");
exitCondition = true;
break;
}
default:{
script.log("Publish to GtS: publishing with current settings canceled.");
exitCondition = true;
break;
}
}
if (exitCondition) return;
}
if (currentParams.created_at){
script.log("Publish to Gts: actual note may have already been published.");
if (!confirmDuplicatePost()){
return;
}
}
if (confirmPublish()){
script.log("Publish to GtS: note publishing confirmed.");
script.tagCurrentNote("Pub2GtS");
} else {
script.log("Publish to GtS: note publishing canceled.");
return;
}

// recover accessToken from persistent variables, if present
let accessToken = script.getPersistentVariable("publishToGts/"+authCode);
// check if accessToken was found on the persistent variables
if (accessToken && accessToken.length > 0){
// access token was found, verifying credentials
credentialsVerified = verifyCredentials(accessToken);
} else {
let registerResponse = request("POST", "/api/v1/apps", "", {"client_name":clientName,"redirect_uris":"urn:ietf:wg:oauth:2.0:oob","scopes":clientMode});
if (!registerResponse){
return;
}
clientId = JSON.parse(registerResponse).client_id;
clientSecret = JSON.parse(registerResponse).client_secret;

// need to show a window with a http link for the user to click
let authAddress = `https://${serverInstance}/oauth/authorize?client_id=${clientId}&redirect_uri=urn:ietf:wg:oauth:2.0:oob&response_type=code&scope=${clientMode}`;
let popupTitle = "GoToSocial Authentication";
let popupText = `<ol><li>Visit <a href="${authAddress}">this link</a></li><li>Paste your code below once authenticated</li></ol>`;
//waiting for user to insert the authorzation code
authCode = script.inputDialogGetText(popupTitle, popupText, "Authorization code here");
if (!authCode){
script.log("No authorization code entered.");
return;
}
script.log("authorization code inserted: " + authCode);
// exchanging for token
let tokenRequest = request("POST", "/oauth/token", "", {"redirect_uri": "urn:ietf:wg:oauth:2.0:oob","client_id": clientId, "client_secret": clientSecret,"grant_type": "authorization_code","code": authCode});
script.log(tokenRequest);
if (!tokenRequest){
script.log("Unable to get access Token.");
return;
} else {
accessToken = JSON.parse(tokenRequest).access_token;
credentialsVerified = verifyCredentials(accessToken);
}
}
if (credentialsVerified){
script.log("Credentials verified!");
//saving accessToken in a persistent variable called with the same name as the authCode
script.setPersistentVariable("publishToGts/"+authCode, accessToken);
} else {
script.log("Credentials verification failed. Check the server or the internet connection and retry!");
return;
}
// We can proceed with posting the actual note
// Getting the current note markdown

let noteSections = current.noteText.split("---");
// posting only if text is present, excluding the Post Header, delimited by ---
if (noteSections.length > 2 && /./gm.test(noteSections[2])){
currentPost = noteSections[2];
} else {
script.log("Publish to GtS: Current note does not have a text to be published.")
return;
}
//populating the status object with status and post parameters
let status = {
"status": currentPost
};

Object.keys(currentParams).forEach(function(param){
status[param] = parseBool(currentParams[param]);
});

let statusResult = JSON.parse(request ("POST", "/api/v1/statuses", accessToken, status));
if (statusResult && statusResult["created_at"]){
additionalParams["created_at"] = statusResult["created_at"];
additionalParams["id"] = statusResult["id"];
additionalParams["url"] = statusResult["url"];
updatePostHeader();
script.tagCurrentNote("Published");
}
return;
}
}
}
Loading

0 comments on commit 819bbd0

Please sign in to comment.