#include "archive_parser.h" #include "src/list_item.h" #include "src/types.h" #include #include #include #include Archive::Archive(QString outbox_filename, ArchiveType archive_type) : outbox_filename(outbox_filename), archive_type(archive_type) {} Archive::InitError Archive::init() { QFile outbox_file(outbox_filename); if (!outbox_file.open(QIODevice::ReadOnly | QIODevice::Text)) return FailedOpeningFile; QJsonParseError json_error; QJsonDocument outbox_json_document = QJsonDocument::fromJson(outbox_file.readAll(), &json_error); outbox_file.close(); if (json_error.error != QJsonParseError::NoError) return JsonParseError; if (outbox_json_document.isEmpty()) return JsonEmpty; if (outbox_json_document.isNull()) return JsonNull; if (outbox_json_document.isObject()) outbox_json = new QJsonObject (outbox_json_document.object()); else return JsonNotObject; // Do some more throughful checks to make sure that the JSON is actually valid. if (not (outbox_json->contains("@context") and outbox_json->value("@context").isString() and (outbox_json->value("@context").toString() == "https://www.w3.org/ns/activitystreams"))) return JsonNotActivityStream; if (outbox_json->contains("orderedItems") and outbox_json->value("orderedItems").isArray()) { outbox_items = new QJsonArray(outbox_json->value("orderedItems").toArray()); // we'll need it during Archive's lifetime } return NoError; } Archive::~Archive() { delete outbox_json; outbox_json = nullptr; delete outbox_items; outbox_items = nullptr; } bool is_status_type_allowed(StatusType status_type, ViewStatusTypes allowed_types) { switch (status_type) { case PUBLIC: return allowed_types.includePublic; case UNLISTED: return allowed_types.includeUnlisted; case PRIVATE: return allowed_types.includePrivate; case DIRECT: return allowed_types.includeDirect; default: return true; } } StatusType get_status_type(QJsonObject obj) { /* * public: * to: #Public * unlisted: * to: /followers * cc: #Public * followers: * to: /followers * direct: * to: the Actors mentioned */ StatusType status_type = UNKNOWN; bool is_private_or_unlisted = false; QJsonArray to; if (obj.value("to").isArray()) { to = obj.value("to").toArray(); // see https://www.w3.org/TR/activitypub/#public-addressing for (auto collection : to) { QString col = collection.toString(); if (col == "https://www.w3.org/ns/activitystreams#Public") { return PUBLIC; // status' privacy can only be promoted to a higher level } else if (col.endsWith("/followers")) // at least for Mastodon… is_private_or_unlisted = true; // private or unlisted } } if (obj.value("cc").isArray()) { QJsonArray cc = obj.value("cc").toArray(); for (auto collection : cc) { QString col = collection.toString(); if (col == "https://www.w3.org/ns/activitystreams#Public" and is_private_or_unlisted) { return UNLISTED; // status' privacy can only be promoted to a higher level } } if (is_private_or_unlisted) return PRIVATE; else if (to.size() > 0) return DIRECT; else if (to.size() == 0 and cc.size() == 0) // sending a direct message to yourself or to another actor that doesn't exist anymore return DIRECT; } return status_type; } void Archive::update_status_list(ViewStatusTypes allowed_types, QListWidget *parent) { int i = 0; for (auto&& item : *outbox_items) { if (item.isObject()){ QJsonObject obj = item.toObject(); if (not obj.value("type").isString()) // this shouldn't happen, but you never know goto next_item; if (obj.value("type").toString() != "Create") // ignoring everything that isn't a Create activity for now goto next_item; // Determine the status' type StatusType status_type = get_status_type(obj); if (not is_status_type_allowed(status_type, allowed_types)) goto next_item; if (obj.value("object").isObject()) { QJsonObject activity = obj.value("object").toObject(); bool has_attachment = false; if (activity.contains("attachment") and not activity["attachment"].toArray().isEmpty()) has_attachment = true; if (allowed_types.onlyWithAttachment and not has_attachment) goto next_item; if (activity.value("content").isString()) { // Strip HTML for display in list, according to https://stackoverflow.com/a/12157835 QTextDocument strip_html; strip_html.setHtml(activity.value("content").toString()); ListItem *item = new ListItem(strip_html.toPlainText(), status_type, has_attachment, parent, i); } } } next_item: ++i; } } QString* get_html_status_info_template() { static QString* html = new QString(); if (html->isNull()) { QFile file("src/templates/status_info.html"); if (!file.open(QIODevice::ReadOnly | QIODevice::Text)) { html->append("Error loading status_info template."); return html; } html->append(file.readAll()); } return html; } QString* get_html_status_info_attachment_template() { static QString* html = new QString(); if (html->isNull()) { QFile file("src/templates/attachment.html"); if (!file.open(QIODevice::ReadOnly | QIODevice::Text)) { html->append("Error loading attachment template."); return html; } html->append(file.readAll()); } return html; } QString get_html_status_object_as_list(QJsonObject obj, QString element_name) { QString text; if (obj.contains(element_name)) { QJsonArray to = obj.value(element_name).toArray(); for (auto collection : to) text.append(collection.toString() + ", "); } if (text.isEmpty()) text = "(empty)"; else text.chop(2); // remove leftover ", " return text; } QString Archive::get_html_status_attachments(QJsonValueRef attachments_ref, int text_zone_width) { QString text; QJsonArray attachments = attachments_ref.toArray(); int i = 1; for (auto attachment_ref : attachments) { QString att_html(*get_html_status_info_attachment_template()); QJsonObject attachment = attachment_ref.toObject(); att_html.replace("{{id}}", QString::number(i)); if (attachment.contains("url")) { QString url = attachment["url"].toString(); QFileInfo path(outbox_filename); url.prepend(path.absolutePath()); att_html.replace("{{path}}", url); att_html.replace("{{filename}}", QFileInfo(url).fileName()); // FIXME: this is ugly } if (attachment.contains("mediaType") and attachment["mediaType"].toString().startsWith("image/")) // dynamically resize image based on the display widget size to avoid horizontal scrolling att_html.replace("{{img-width}}", QString::number((float)text_zone_width - (float)text_zone_width*0.15)); else att_html.replace("{{img-width}}", "0"); if (attachment.contains("name")) att_html.replace("{{alt-text}}", attachment["name"].toString()); text.append(att_html); ++i; } return text; } // status_index is assumed to be a valid index QString Archive::get_html_status_info(int status_index, int text_zone_width) { QString info_text(*get_html_status_info_template()); QJsonObject obj = outbox_items->at(status_index).toObject(); if (obj.contains("type") and obj["type"].isString()) info_text.replace("{{type}}", obj["type"].toString()); if (obj.contains("published") and obj["published"].isString()) info_text.replace("{{published}}", obj["published"].toString()); if (obj.contains("id") and obj["id"].isString()) info_text.replace("{{url-id}}", obj["id"].toString()); info_text.replace("{{to}}", get_html_status_object_as_list(obj, "to")); info_text.replace("{{cc}}", get_html_status_object_as_list(obj, "cc")); if (obj["object"].isObject()) { QJsonObject activity = obj["object"].toObject(); if (activity.contains("summary")) { QString summary_text = activity["summary"].toString(); if (summary_text.isEmpty()) summary_text = "(empty)"; info_text.replace("{{summary}}", summary_text); } info_text.replace("{{attachment-div}}", get_html_status_attachments(activity["attachment"], text_zone_width)); if (activity.contains("url") and activity["url"].isString()) info_text.replace("{{url-status}}", activity["url"].toString()); if (activity["content"].isString()) { info_text.replace("{{content}}", activity["content"].toString()); } } return info_text; } QString Archive::get_html_status_text(int status_index) { QString text(""); QJsonObject obj = outbox_items->at(status_index).toObject(); if (obj["object"].isObject()) { QJsonObject activity = obj["object"].toObject(); if (activity.contains("summary")) { QString summary_text = activity["summary"].toString(); if (not summary_text.isEmpty()) text.append(QString("

CW: {{summary}}

").replace("{{summary}}", summary_text)); } if (activity["content"].isString()) { text.append(QString("
{{content}}
").replace("{{content}}", activity["content"].toString())); } } return text; }