diff --git a/discourse/README b/discourse/README
new file mode 100644
index 00000000..338b10bf
--- /dev/null
+++ b/discourse/README
@@ -0,0 +1,28 @@
+Discourse connector
+===================
+
+The Discourse connectors detects incoming mails from Discourse and
+improves them by fetching the content and user data via API.
+
+Prerequisites
+-------------
+The user has to configure the mail interface so that the user's mails
+can be fetched via Friendica. Then the user has to activate the
+mailing list mode in Discourse.
+
+The mailing list mode in Discourse knows two different options:
+1. Get all posts - including your own. This will create duplicates
+ if you post via Friendica.
+2. Don't get your own posts. Then you will missing all your posts
+ that you made directly on Discourse. Since you cannot create
+ a new post via this connector (only comments are possible)
+ this is not a good choice either.
+
+Known problems
+--------------
+- You can't create starting posts
+- Either you don't get your own posts you made directly on Discourse
+ or you do get duplicates for every post you made via Friendica.
+- Non public categories are currently only working via some workaround
+ without the API, which most likely will cause some content problems.
+- links to Discourse profiles in the posts are invalid.
diff --git a/discourse/discourse.php b/discourse/discourse.php
new file mode 100644
index 00000000..262ee554
--- /dev/null
+++ b/discourse/discourse.php
@@ -0,0 +1,339 @@
+
+ *
+ */
+use Friendica\App;
+use Friendica\Core\Hook;
+use Friendica\Core\L10n;
+use Friendica\Core\Logger;
+use Friendica\Core\PConfig;
+use Friendica\Core\Renderer;
+use Friendica\Core\Protocol;
+use Friendica\Database\DBA;
+use Friendica\Model\Contact;
+use Friendica\Util\XML;
+use Friendica\Content\Text\Markdown;
+use Friendica\Util\Network;
+use Friendica\Util\Strings;
+Use Friendica\Util\DateTimeFormat;
+
+/* Todo:
+ * - Obtaining API tokens to be able to read non public posts as well
+ * - Handling duplicates (possibly using some non visible marker)
+ * - Fetching missing posts
+ * - Fetch topic information
+ * - Support mail free mode when write tokens are available
+ * - Fix incomplete (relative) links (hosts are missing)
+*/
+
+function discourse_install()
+{
+ Hook::register('email_getmessage', __FILE__, 'discourse_email_getmessage');
+ Hook::register('connector_settings', __FILE__, 'discourse_settings');
+ Hook::register('connector_settings_post', __FILE__, 'discourse_settings_post');
+}
+
+function discourse_settings(App $a, &$s)
+{
+ if (!local_user()) {
+ return;
+ }
+
+ $enabled = intval(PConfig::get(local_user(), 'discourse', 'enabled'));
+
+ $t = Renderer::getMarkupTemplate('settings.tpl', 'addon/discourse/');
+ $s .= Renderer::replaceMacros($t, [
+ '$title' => L10n::t('Discourse'),
+ '$enabled' => ['enabled', L10n::t('Enable processing of Discourse mailing list mails'), $enabled, L10n::t('If enabled, incoming mails from Discourse will be improved so they look much better. To make it work, you have to configure the e-mail settings in Friendica. You also have to enable the mailing list mode in Discourse. Then you have to add the Discourse mail account as contact.')],
+ '$submit' => L10n::t('Save Settings'),
+ ]);
+}
+
+function discourse_settings_post(App $a)
+{
+ if (!local_user() || empty($_POST['discourse-submit'])) {
+ return;
+ }
+
+ PConfig::set(local_user(), 'discourse', 'enabled', intval($_POST['enabled']));
+}
+
+function discourse_email_getmessage(App $a, &$message)
+{
+ if (empty($message['item']['uid'])) {
+ return;
+ }
+
+ if (!PConfig::get($message['item']['uid'], 'discourse', 'enabled')) {
+ return;
+ }
+
+ // We do assume that all Discourse servers are running with SSL
+ if (preg_match('=topic/(.*\d)/(.*\d)@(.*)=', $message['item']['uri'], $matches) &&
+ discourse_fetch_post_from_api($message, $matches[2], $matches[3])) {
+ Logger::info('Fetched comment via API (message-id mode)', ['host' => $matches[3], 'topic' => $matches[1], 'post' => $matches[2]]);
+ return;
+ }
+
+ if (preg_match('=topic/(.*\d)@(.*)=', $message['item']['uri'], $matches) &&
+ discourse_fetch_topic_from_api($message, 'https://' . $matches[2], $matches[1], 1)) {
+ Logger::info('Fetched starting post via API (message-id mode)', ['host' => $matches[2], 'topic' => $matches[1]]);
+ return;
+ }
+
+ // Search in the text part for the link to the discourse entry and the text body
+ if (!empty($message['text'])) {
+ $message = discourse_get_text($message);
+ }
+
+ if (empty($message['item']['plink']) || !preg_match('=(http.*)/t/.*/(.*\d)/(.*\d)=', $message['item']['plink'], $matches)) {
+ Logger::info('This is no Discourse post');
+ return;
+ }
+
+ if (discourse_fetch_topic_from_api($message, $matches[1], $matches[2], $matches[3])) {
+ Logger::info('Fetched post via API (plink mode)', ['host' => $matches[1], 'topic' => $matches[2], 'id' => $matches[3]]);
+ return;
+ }
+
+ Logger::info('Fallback mode', ['plink' => $message['item']['plink']]);
+ // Search in the HTML part for the discourse entry and the author profile
+ if (!empty($message['html'])) {
+ $message = discourse_get_html($message);
+ }
+
+ // Remove the title on comments, they don't serve any purpose there
+ if ($message['item']['parent-uri'] != $message['item']['uri']) {
+ unset($message['item']['title']);
+ }
+}
+
+function discourse_fetch_post($host, $topic, $pid)
+{
+ $url = $host . '/t/' . $topic . '/' . $pid . '.json';
+ $curlResult = Network::curl($url);
+ if (!$curlResult->isSuccess()) {
+ Logger::info('No success', ['url' => $url]);
+ return false;
+ }
+
+ $raw = $curlResult->getBody();
+ $data = json_decode($raw, true);
+ $posts = $data['post_stream']['posts'];
+ foreach($posts as $post) {
+ if ($post['post_number'] != $pid) {
+ /// @todo Possibly fetch missing posts here
+ continue;
+ }
+ Logger::info('Got post data from topic', $post);
+ return $post;
+ }
+
+ Logger::info('Post not found', ['host' => $host, 'topic' => $topic, 'pid' => $pid]);
+ return false;
+}
+
+function discourse_fetch_topic_from_api(&$message, $host, $topic, $pid)
+{
+ $post = discourse_fetch_post($host, $topic, $pid);
+ if (empty($post)) {
+ return false;
+ }
+
+ $message = discourse_process_post($message, $post, $host);
+ return true;
+}
+
+function discourse_fetch_post_from_api(&$message, $post, $host)
+{
+ $hostaddr = 'https://' . $host;
+ $url = $hostaddr . '/posts/' . $post . '.json';
+ $curlResult = Network::curl($url);
+ if (!$curlResult->isSuccess()) {
+ return false;
+ }
+
+ $raw = $curlResult->getBody();
+ $data = json_decode($raw, true);
+ if (empty($data)) {
+ return false;
+ }
+
+ $message = discourse_process_post($message, $data, $hostaddr);
+
+ Logger::info('Got API data', $message);
+ return true;
+}
+
+function discourse_get_user($post, $hostaddr)
+{
+ $host = parse_url($hostaddr, PHP_URL_HOST);
+
+ // Currently unused contact fields:
+ // - display_username
+ // - user_id
+
+ $contact = [];
+ $contact['uid'] = 0;
+ $contact['network'] = Protocol::DISCOURSE;
+ $contact['name'] = $contact['nick'] = $post['username'];
+ if (!empty($post['name'])) {
+ $contact['name'] = $post['name'];
+ }
+
+ $contact['about'] = $post['user_title'];
+
+ if (parse_url($post['avatar_template'], PHP_URL_SCHEME)) {
+ $contact['photo'] = str_replace('{size}', '300', $post['avatar_template']);
+ } else {
+ $contact['photo'] = $hostaddr . str_replace('{size}', '300', $post['avatar_template']);
+ }
+
+ $contact['addr'] = $contact['nick'] . '@' . $host;
+ $contact['contact-type'] = Contact::TYPE_PERSON;
+ $contact['url'] = $hostaddr . '/u/' . $contact['nick'];
+ $contact['nurl'] = Strings::normaliseLink($contact['url']);
+ $contact['baseurl'] = $hostaddr;
+ Logger::info('Contact', $contact);
+ $contact['id'] = Contact::getIdForURL($contact['url'], 0, true, $contact);
+ if (!empty($contact['id'])) {
+ $avatar = $contact['photo'];
+ unset($contact['photo']);
+ DBA::update('contact', $contact, ['id' => $contact['id']]);
+ Contact::updateAvatar($avatar, 0, $contact['id']);
+ $contact['photo'] = $avatar;
+ }
+
+ return $contact;
+}
+
+function discourse_process_post($message, $post, $hostaddr)
+{
+ $host = parse_url($hostaddr, PHP_URL_HOST);
+
+ $message['html'] = $post['cooked'];
+
+ $contact = discourse_get_user($post, $hostaddr);
+ $message['item']['author-id'] = $contact['id'];
+ $message['item']['author-link'] = $contact['url'];
+ $message['item']['author-name'] = $contact['name'];
+ $message['item']['author-avatar'] = $contact['photo'];
+ $message['item']['created'] = DateTimeFormat::utc($post['created_at']);
+ $message['item']['plink'] = $hostaddr . '/t/' . $post['topic_slug'] . '/' . $post['topic_id'] . '/' . $post['post_number'];
+
+ if ($post['post_number'] == 1) {
+ $message['item']['parent-uri'] = $message['item']['uri'] = 'topic/' . $post['topic_id'] . '@' . $host;
+
+ // Remove the Discourse forum name from the subject
+ $pattern = '=\[.*\].*\s(\[.*\].*)=';
+ if (preg_match($pattern, $message['item']['title'])) {
+ $message['item']['title'] = preg_replace($pattern, '$1', $message['item']['title']);
+ }
+ /// @ToDo Fetch thread information
+ } else {
+ $message['item']['uri'] = 'topic/' . $post['topic_id'] . '/' . $post['id'] . '@' . $host;
+ unset($message['item']['title']);
+ if (empty($post['reply_to_post_number']) || $post['reply_to_post_number'] == 1) {
+ $message['item']['parent-uri'] = 'topic/' . $post['topic_id'] . '@' . $host;
+ } else {
+ $reply = discourse_fetch_post($hostaddr, $post['topic_id'], $post['reply_to_post_number']);
+ $message['item']['parent-uri'] = 'topic/' . $post['topic_id'] . '/' . $reply['id'] . '@' . $host;
+ }
+ }
+
+ return $message;
+}
+
+function discourse_get_html($message)
+{
+ $doc = new DOMDocument();
+ $doc2 = new DOMDocument();
+ $doc->preserveWhiteSpace = false;
+
+ $html = mb_convert_encoding($message['html'], 'HTML-ENTITIES', "UTF-8");
+ @$doc->loadHTML($html, LIBXML_HTML_NODEFDTD);
+
+ $xpath = new DomXPath($doc);
+
+ // Fetch the first 'div' before the 'hr' - hopefully this fits for all systems
+ $result = $xpath->query("//hr//preceding::div[1]");
+ $div = $doc2->importNode($result->item(0), true);
+ $doc2->appendChild($div);
+ $message['html'] = $doc2->saveHTML();
+ Logger::info('Found html body', ['html' => $message['html']]);
+
+ $profile = discourse_get_profile($xpath);
+ if (!empty($profile['url'])) {
+ Logger::info('Found profile', $profile);
+ $message['item']['author-id'] = Contact::getIdForURL($profile['url'], 0, true, $profile);
+ $message['item']['author-link'] = $profile['url'];
+ $message['item']['author-name'] = $profile['name'];
+ $message['item']['author-avatar'] = $profile['photo'];
+ }
+
+ return $message;
+}
+
+function discourse_get_text($message)
+{
+ $text = $message['text'];
+ $text = str_replace("\r", '', $text);
+ $pos = strpos($text, "\n---\n");
+ if ($pos == 0) {
+ Logger::info('No separator found', ['text' => $text]);
+ return $message;
+ }
+
+ $message['text'] = trim(substr($text, 0, $pos));
+
+ Logger::info('Found text body', ['text' => $message['text']]);
+
+ $message['text'] = Markdown::toBBCode($message['text']);
+
+ $text = substr($text, $pos);
+ Logger::info('Found footer', ['text' => $text]);
+ if (preg_match('=\((http.*/t/.*/.*\d/.*\d)\)=', $text, $link)) {
+ $message['item']['plink'] = $link[1];
+ Logger::info('Found plink', ['plink' => $message['item']['plink']]);
+ }
+ return $message;
+}
+
+function discourse_get_profile($xpath)
+{
+ $profile = [];
+ $list = $xpath->query("//td//following::img");
+ foreach ($list as $node) {
+ $attr = [];
+ foreach ($node->attributes as $attribute) {
+ $attr[$attribute->name] = $attribute->value;
+ }
+
+ if (!empty($attr['src']) && !empty($attr['title'])
+ && !empty($attr['width']) && !empty($attr['height'])
+ && ($attr['width'] == $attr['height'])) {
+ $profile = ['photo' => $attr['src'], 'name' => $attr['title']];
+ break;
+ }
+ }
+
+ $list = $xpath->query("//td//following::a");
+ foreach ($list as $node) {
+ if (!empty(trim($node->textContent)) && $node->attributes->length) {
+ $attr = [];
+ foreach ($node->attributes as $attribute) {
+ $attr[$attribute->name] = $attribute->value;
+ }
+ if (!empty($attr['href']) && (strpos($attr['href'], '/' . $profile['name']))) {
+ $profile['url'] = $attr['href'];
+ break;
+ }
+ }
+ }
+ return $profile;
+}
diff --git a/discourse/templates/settings.tpl b/discourse/templates/settings.tpl
new file mode 100644
index 00000000..d4b0bf8c
--- /dev/null
+++ b/discourse/templates/settings.tpl
@@ -0,0 +1,15 @@
+
+ {{$title}}
+
+