2019-03-29 01:53:04 +00:00
|
|
|
<?php
|
|
|
|
/** @license MIT
|
|
|
|
* Copyright 2017 J. King, Dustin Wilson et al.
|
|
|
|
* See LICENSE and AUTHORS files for details */
|
|
|
|
|
|
|
|
declare(strict_types=1);
|
|
|
|
namespace JKingWeb\Arsse\ImportExport;
|
|
|
|
|
|
|
|
use JKingWeb\Arsse\Arsse;
|
|
|
|
|
|
|
|
class OPML {
|
|
|
|
public function export(string $user, bool $flat = false): string {
|
2019-03-29 13:02:39 +00:00
|
|
|
$tags = [];
|
2019-03-29 01:53:04 +00:00
|
|
|
$folders = [];
|
|
|
|
$parents = [0 => null];
|
2019-03-29 13:02:39 +00:00
|
|
|
// create a base document
|
2019-03-29 01:53:04 +00:00
|
|
|
$document = new \DOMDocument("1.0", "utf-8");
|
|
|
|
$document->formatOutput = true;
|
|
|
|
$document->appendChild($document->createElement("opml"));
|
|
|
|
$document->documentElement->setAttribute("version", "2.0");
|
|
|
|
$document->documentElement->appendChild($document->createElement("head"));
|
|
|
|
// create the "root folder" node (the body node, in OPML terms)
|
|
|
|
$folders[0] = $document->createElement("body");
|
2019-03-29 13:02:39 +00:00
|
|
|
// begin a transaction for read isolation
|
2019-03-29 01:53:04 +00:00
|
|
|
$transaction = Arsse::$db->begin();
|
2019-03-29 13:02:39 +00:00
|
|
|
// gather up the list of tags for each subscription
|
2019-03-29 01:53:04 +00:00
|
|
|
foreach (Arsse::$db->tagSummarize($user) as $r) {
|
|
|
|
$sub = $r['subscription'];
|
|
|
|
$tag = $r['name'];
|
2019-03-29 13:02:39 +00:00
|
|
|
// strip out any commas in the tag name; sadly this is lossy as OPML has no escape mechanism
|
2019-03-29 01:53:04 +00:00
|
|
|
$tag = str_replace(",", "", $tag);
|
|
|
|
if (!isset($tags[$sub])) {
|
|
|
|
$tags[$sub] = [];
|
|
|
|
}
|
|
|
|
$tags[$sub][] = $tag;
|
|
|
|
}
|
|
|
|
if (!$flat) {
|
2019-03-29 13:02:39 +00:00
|
|
|
// unless the output is requested flat, gather up the list of folders, using their database IDs as array indices
|
2019-03-29 01:53:04 +00:00
|
|
|
foreach (Arsse::$db->folderList($user) as $r) {
|
2019-03-29 13:02:39 +00:00
|
|
|
// note the index of its parent folder for later tree construction
|
2019-03-29 01:53:04 +00:00
|
|
|
$parents[$r['id']] = $r['parent'] ?? 0;
|
2019-03-29 13:02:39 +00:00
|
|
|
// create a DOM node for each folder; we don't insert it yet
|
2019-03-29 01:53:04 +00:00
|
|
|
$el = $document->createElement("outline");
|
|
|
|
$el->setAttribute("text", $r['name']);
|
|
|
|
$folders[$r['id']] = $el;
|
|
|
|
}
|
2019-03-30 14:01:12 +00:00
|
|
|
}
|
|
|
|
// insert each folder into its parent node; for the root folder the parent is the document root node
|
|
|
|
foreach ($folders as $id => $el) {
|
|
|
|
$parent = $folders[$parents[$id]] ?? $document->documentElement;
|
|
|
|
$parent->appendChild($el);
|
2019-03-29 01:53:04 +00:00
|
|
|
}
|
2019-03-29 13:02:39 +00:00
|
|
|
// create a DOM node for each subscription and insert them directly into their folder DOM node
|
2019-03-29 01:53:04 +00:00
|
|
|
foreach (Arsse::$db->subscriptionList($user) as $r) {
|
|
|
|
$el = $document->createElement(("outline"));
|
|
|
|
$el->setAttribute("type", "rss");
|
2019-03-30 14:01:12 +00:00
|
|
|
$el->setAttribute("text", $r['title']);
|
2019-03-29 01:53:04 +00:00
|
|
|
$el->setAttribute("xmlUrl", $r['url']);
|
2019-03-29 13:02:39 +00:00
|
|
|
// include the category attribute only if there are tags
|
2019-03-30 14:01:12 +00:00
|
|
|
if (isset($tags[$r['id']]) && sizeof($tags[$r['id']])) {
|
2019-03-29 01:53:04 +00:00
|
|
|
$el->setAttribute("category", implode(",", $tags[$r['id']]));
|
|
|
|
}
|
2019-03-29 13:02:39 +00:00
|
|
|
// if flat output was requested subscriptions are inserted into the root folder
|
2019-03-29 01:53:04 +00:00
|
|
|
($folders[$r['folder'] ?? 0] ?? $folders[0])->appendChild($el);
|
|
|
|
}
|
2019-03-29 13:02:39 +00:00
|
|
|
// release the transaction
|
2019-03-29 01:53:04 +00:00
|
|
|
$transaction->rollback();
|
|
|
|
// return the serialization
|
|
|
|
return $document->saveXML();
|
|
|
|
}
|
|
|
|
}
|