summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--README.markdown1
-rw-r--r--app/Controller/Webhook.php16
-rw-r--r--app/Integration/PostmarkWebhook.php67
-rw-r--r--app/Locale/da_DK/translations.php4
-rw-r--r--app/Locale/de_DE/translations.php4
-rw-r--r--app/Locale/es_ES/translations.php4
-rw-r--r--app/Locale/fi_FI/translations.php4
-rw-r--r--app/Locale/fr_FR/translations.php4
-rw-r--r--app/Locale/hu_HU/translations.php4
-rw-r--r--app/Locale/it_IT/translations.php4
-rw-r--r--app/Locale/ja_JP/translations.php4
-rw-r--r--app/Locale/nl_NL/translations.php4
-rw-r--r--app/Locale/pl_PL/translations.php4
-rw-r--r--app/Locale/pt_BR/translations.php4
-rw-r--r--app/Locale/ru_RU/translations.php4
-rw-r--r--app/Locale/sr_Latn_RS/translations.php4
-rw-r--r--app/Locale/sv_SE/translations.php4
-rw-r--r--app/Locale/th_TH/translations.php4
-rw-r--r--app/Locale/tr_TR/translations.php4
-rw-r--r--app/Locale/zh_CN/translations.php4
-rw-r--r--app/Model/Project.php31
-rw-r--r--app/Model/User.php12
-rw-r--r--app/Schema/Mysql.php7
-rw-r--r--app/Schema/Postgres.php7
-rw-r--r--app/Schema/Sqlite.php7
-rw-r--r--app/ServiceProvider/ClassProvider.php1
-rw-r--r--app/Template/config/integrations.php8
-rw-r--r--app/Template/project/edit.php4
-rw-r--r--app/Template/project/index.php4
-rw-r--r--assets/img/gravatar-icon.pngbin0 -> 517 bytes
-rw-r--r--assets/img/postmark-icon.pngbin0 -> 474 bytes
-rw-r--r--composer.json3
-rw-r--r--composer.lock48
-rw-r--r--docs/postmark.markdown59
-rw-r--r--tests/units/Base.php5
-rw-r--r--tests/units/PostmarkWebhookTest.php83
-rw-r--r--tests/units/ProjectTest.php53
37 files changed, 478 insertions, 6 deletions
diff --git a/README.markdown b/README.markdown
index 5bf71088..4c4eda67 100644
--- a/README.markdown
+++ b/README.markdown
@@ -108,6 +108,7 @@ Documentation
- [Gitlab webhooks](docs/gitlab-webhooks.markdown)
- [Hipchat](docs/hipchat.markdown)
- [Slack](docs/slack.markdown)
+- [Postmark](docs/postmark.markdown)
#### More
diff --git a/app/Controller/Webhook.php b/app/Controller/Webhook.php
index ef79379f..afa0543a 100644
--- a/app/Controller/Webhook.php
+++ b/app/Controller/Webhook.php
@@ -100,4 +100,20 @@ class Webhook extends Base
echo $result ? 'PARSED' : 'IGNORED';
}
+
+ /**
+ * Handle Postmark webhooks
+ *
+ * @access public
+ */
+ public function postmark()
+ {
+ if ($this->config->get('webhook_token') !== $this->request->getStringParam('token')) {
+ $this->response->text('Not Authorized', 401);
+ }
+
+ $result = $this->postmarkWebhook->parsePayload($this->request->getJson() ?: array());
+
+ echo $result ? 'PARSED' : 'IGNORED';
+ }
}
diff --git a/app/Integration/PostmarkWebhook.php b/app/Integration/PostmarkWebhook.php
new file mode 100644
index 00000000..6387ba20
--- /dev/null
+++ b/app/Integration/PostmarkWebhook.php
@@ -0,0 +1,67 @@
+<?php
+
+namespace Integration;
+
+use HTML_To_Markdown;
+
+/**
+ * Postmark Webhook
+ *
+ * @package integration
+ * @author Frederic Guillot
+ */
+class PostmarkWebhook extends Base
+{
+ /**
+ * Parse incoming email
+ *
+ * @access public
+ * @param array $payload Incoming email
+ * @return boolean
+ */
+ public function parsePayload(array $payload)
+ {
+ if (empty($payload['From']) || empty($payload['Subject']) || empty($payload['MailboxHash']) || empty($payload['TextBody'])) {
+ return false;
+ }
+
+ // The user must exists in Kanboard
+ $user = $this->user->getByEmail($payload['From']);
+
+ if (empty($user)) {
+ $this->container['logger']->debug('PostmarkWebhook: ignored => user not found');
+ return false;
+ }
+
+ // The project must have a short name
+ $project = $this->project->getByIdentifier($payload['MailboxHash']);
+
+ if (empty($project)) {
+ $this->container['logger']->debug('PostmarkWebhook: ignored => project not found');
+ return false;
+ }
+
+ // The user must be member of the project
+ if (! $this->projectPermission->isMember($project['id'], $user['id'])) {
+ $this->container['logger']->debug('PostmarkWebhook: ignored => user is not member of the project');
+ return false;
+ }
+
+ // Get the Markdown contents
+ if (empty($payload['HtmlBody'])) {
+ $description = $payload['TextBody'];
+ }
+ else {
+ $markdown = new HTML_To_Markdown($payload['HtmlBody'], array('strip_tags' => true));
+ $description = $markdown->output();
+ }
+
+ // Finally, we create the task
+ return (bool) $this->taskCreation->create(array(
+ 'project_id' => $project['id'],
+ 'title' => $payload['Subject'],
+ 'description' => $description,
+ 'creator_id' => $user['id'],
+ ));
+ }
+}
diff --git a/app/Locale/da_DK/translations.php b/app/Locale/da_DK/translations.php
index cc602ef4..f021ea34 100644
--- a/app/Locale/da_DK/translations.php
+++ b/app/Locale/da_DK/translations.php
@@ -858,4 +858,8 @@ return array(
// 'Take a screenshot and press CTRL+V or ⌘+V to paste here.' => '',
// 'Screenshot uploaded successfully.' => '',
// 'SEK - Swedish Krona' => '',
+ // 'The project identifier is an optional alphanumeric code used to identify your project.' => '',
+ // 'Identifier' => '',
+ // 'Postmark (incoming emails)' => '',
+ // 'Help on Postmark integration' => '',
);
diff --git a/app/Locale/de_DE/translations.php b/app/Locale/de_DE/translations.php
index 6a589cdd..f58e4630 100644
--- a/app/Locale/de_DE/translations.php
+++ b/app/Locale/de_DE/translations.php
@@ -858,4 +858,8 @@ return array(
// 'Take a screenshot and press CTRL+V or ⌘+V to paste here.' => '',
// 'Screenshot uploaded successfully.' => '',
// 'SEK - Swedish Krona' => '',
+ // 'The project identifier is an optional alphanumeric code used to identify your project.' => '',
+ // 'Identifier' => '',
+ // 'Postmark (incoming emails)' => '',
+ // 'Help on Postmark integration' => '',
);
diff --git a/app/Locale/es_ES/translations.php b/app/Locale/es_ES/translations.php
index d230d845..3160e6f6 100644
--- a/app/Locale/es_ES/translations.php
+++ b/app/Locale/es_ES/translations.php
@@ -858,4 +858,8 @@ return array(
// 'Take a screenshot and press CTRL+V or ⌘+V to paste here.' => '',
// 'Screenshot uploaded successfully.' => '',
// 'SEK - Swedish Krona' => '',
+ // 'The project identifier is an optional alphanumeric code used to identify your project.' => '',
+ // 'Identifier' => '',
+ // 'Postmark (incoming emails)' => '',
+ // 'Help on Postmark integration' => '',
);
diff --git a/app/Locale/fi_FI/translations.php b/app/Locale/fi_FI/translations.php
index badb7d88..934f90f4 100644
--- a/app/Locale/fi_FI/translations.php
+++ b/app/Locale/fi_FI/translations.php
@@ -858,4 +858,8 @@ return array(
// 'Take a screenshot and press CTRL+V or ⌘+V to paste here.' => '',
// 'Screenshot uploaded successfully.' => '',
// 'SEK - Swedish Krona' => '',
+ // 'The project identifier is an optional alphanumeric code used to identify your project.' => '',
+ // 'Identifier' => '',
+ // 'Postmark (incoming emails)' => '',
+ // 'Help on Postmark integration' => '',
);
diff --git a/app/Locale/fr_FR/translations.php b/app/Locale/fr_FR/translations.php
index 3bc3e7df..cdb9045b 100644
--- a/app/Locale/fr_FR/translations.php
+++ b/app/Locale/fr_FR/translations.php
@@ -860,4 +860,8 @@ return array(
'Take a screenshot and press CTRL+V or ⌘+V to paste here.' => 'Prenez une capture d\'écran et appuyez sur CTRL+V ou ⌘+V pour coller ici.',
'Screenshot uploaded successfully.' => 'Capture d\'écran téléchargée avec succès.',
'SEK - Swedish Krona' => 'SEK - Couronne suédoise',
+ 'The project identifier is an optional alphanumeric code used to identify your project.' => 'L\'identificateur du projet est un code alpha-numérique optionnel pour identifier votre projet.',
+ 'Identifier' => 'Identificateur',
+ 'Postmark (incoming emails)' => 'Postmark (emails entrants)',
+ 'Help on Postmark integration' => 'Aide sur l\'intégration avec Postmark',
);
diff --git a/app/Locale/hu_HU/translations.php b/app/Locale/hu_HU/translations.php
index 558bb25e..38bed1a2 100644
--- a/app/Locale/hu_HU/translations.php
+++ b/app/Locale/hu_HU/translations.php
@@ -858,4 +858,8 @@ return array(
// 'Take a screenshot and press CTRL+V or ⌘+V to paste here.' => '',
// 'Screenshot uploaded successfully.' => '',
// 'SEK - Swedish Krona' => '',
+ // 'The project identifier is an optional alphanumeric code used to identify your project.' => '',
+ // 'Identifier' => '',
+ // 'Postmark (incoming emails)' => '',
+ // 'Help on Postmark integration' => '',
);
diff --git a/app/Locale/it_IT/translations.php b/app/Locale/it_IT/translations.php
index c8df81ab..59f057b3 100644
--- a/app/Locale/it_IT/translations.php
+++ b/app/Locale/it_IT/translations.php
@@ -858,4 +858,8 @@ return array(
// 'Take a screenshot and press CTRL+V or ⌘+V to paste here.' => '',
// 'Screenshot uploaded successfully.' => '',
// 'SEK - Swedish Krona' => '',
+ // 'The project identifier is an optional alphanumeric code used to identify your project.' => '',
+ // 'Identifier' => '',
+ // 'Postmark (incoming emails)' => '',
+ // 'Help on Postmark integration' => '',
);
diff --git a/app/Locale/ja_JP/translations.php b/app/Locale/ja_JP/translations.php
index 11af3feb..9fa0a723 100644
--- a/app/Locale/ja_JP/translations.php
+++ b/app/Locale/ja_JP/translations.php
@@ -858,4 +858,8 @@ return array(
// 'Take a screenshot and press CTRL+V or ⌘+V to paste here.' => '',
// 'Screenshot uploaded successfully.' => '',
// 'SEK - Swedish Krona' => '',
+ // 'The project identifier is an optional alphanumeric code used to identify your project.' => '',
+ // 'Identifier' => '',
+ // 'Postmark (incoming emails)' => '',
+ // 'Help on Postmark integration' => '',
);
diff --git a/app/Locale/nl_NL/translations.php b/app/Locale/nl_NL/translations.php
index 37c7c69d..5f57937c 100644
--- a/app/Locale/nl_NL/translations.php
+++ b/app/Locale/nl_NL/translations.php
@@ -858,4 +858,8 @@ return array(
// 'Take a screenshot and press CTRL+V or ⌘+V to paste here.' => '',
// 'Screenshot uploaded successfully.' => '',
// 'SEK - Swedish Krona' => '',
+ // 'The project identifier is an optional alphanumeric code used to identify your project.' => '',
+ // 'Identifier' => '',
+ // 'Postmark (incoming emails)' => '',
+ // 'Help on Postmark integration' => '',
);
diff --git a/app/Locale/pl_PL/translations.php b/app/Locale/pl_PL/translations.php
index 366c6371..42b2865e 100644
--- a/app/Locale/pl_PL/translations.php
+++ b/app/Locale/pl_PL/translations.php
@@ -858,4 +858,8 @@ return array(
// 'Take a screenshot and press CTRL+V or ⌘+V to paste here.' => '',
// 'Screenshot uploaded successfully.' => '',
// 'SEK - Swedish Krona' => '',
+ // 'The project identifier is an optional alphanumeric code used to identify your project.' => '',
+ // 'Identifier' => '',
+ // 'Postmark (incoming emails)' => '',
+ // 'Help on Postmark integration' => '',
);
diff --git a/app/Locale/pt_BR/translations.php b/app/Locale/pt_BR/translations.php
index 87845f77..4dd237ec 100644
--- a/app/Locale/pt_BR/translations.php
+++ b/app/Locale/pt_BR/translations.php
@@ -858,4 +858,8 @@ return array(
// 'Take a screenshot and press CTRL+V or ⌘+V to paste here.' => '',
// 'Screenshot uploaded successfully.' => '',
// 'SEK - Swedish Krona' => '',
+ // 'The project identifier is an optional alphanumeric code used to identify your project.' => '',
+ // 'Identifier' => '',
+ // 'Postmark (incoming emails)' => '',
+ // 'Help on Postmark integration' => '',
);
diff --git a/app/Locale/ru_RU/translations.php b/app/Locale/ru_RU/translations.php
index 18fd6dfe..e3039db9 100644
--- a/app/Locale/ru_RU/translations.php
+++ b/app/Locale/ru_RU/translations.php
@@ -858,4 +858,8 @@ return array(
// 'Take a screenshot and press CTRL+V or ⌘+V to paste here.' => '',
// 'Screenshot uploaded successfully.' => '',
// 'SEK - Swedish Krona' => '',
+ // 'The project identifier is an optional alphanumeric code used to identify your project.' => '',
+ // 'Identifier' => '',
+ // 'Postmark (incoming emails)' => '',
+ // 'Help on Postmark integration' => '',
);
diff --git a/app/Locale/sr_Latn_RS/translations.php b/app/Locale/sr_Latn_RS/translations.php
index cb2d7287..6e4a8228 100644
--- a/app/Locale/sr_Latn_RS/translations.php
+++ b/app/Locale/sr_Latn_RS/translations.php
@@ -858,4 +858,8 @@ return array(
// 'Take a screenshot and press CTRL+V or ⌘+V to paste here.' => '',
// 'Screenshot uploaded successfully.' => '',
// 'SEK - Swedish Krona' => '',
+ // 'The project identifier is an optional alphanumeric code used to identify your project.' => '',
+ // 'Identifier' => '',
+ // 'Postmark (incoming emails)' => '',
+ // 'Help on Postmark integration' => '',
);
diff --git a/app/Locale/sv_SE/translations.php b/app/Locale/sv_SE/translations.php
index e67402b2..abdb2b70 100644
--- a/app/Locale/sv_SE/translations.php
+++ b/app/Locale/sv_SE/translations.php
@@ -858,4 +858,8 @@ return array(
// 'Take a screenshot and press CTRL+V or ⌘+V to paste here.' => '',
// 'Screenshot uploaded successfully.' => '',
'SEK - Swedish Krona' => 'SEK - Svensk Krona',
+ // 'The project identifier is an optional alphanumeric code used to identify your project.' => '',
+ // 'Identifier' => '',
+ // 'Postmark (incoming emails)' => '',
+ // 'Help on Postmark integration' => '',
);
diff --git a/app/Locale/th_TH/translations.php b/app/Locale/th_TH/translations.php
index 024b211a..b6c140d6 100644
--- a/app/Locale/th_TH/translations.php
+++ b/app/Locale/th_TH/translations.php
@@ -858,4 +858,8 @@ return array(
// 'Take a screenshot and press CTRL+V or ⌘+V to paste here.' => '',
// 'Screenshot uploaded successfully.' => '',
// 'SEK - Swedish Krona' => '',
+ // 'The project identifier is an optional alphanumeric code used to identify your project.' => '',
+ // 'Identifier' => '',
+ // 'Postmark (incoming emails)' => '',
+ // 'Help on Postmark integration' => '',
);
diff --git a/app/Locale/tr_TR/translations.php b/app/Locale/tr_TR/translations.php
index c5c1c79e..7457e0b1 100644
--- a/app/Locale/tr_TR/translations.php
+++ b/app/Locale/tr_TR/translations.php
@@ -858,4 +858,8 @@ return array(
// 'Take a screenshot and press CTRL+V or ⌘+V to paste here.' => '',
// 'Screenshot uploaded successfully.' => '',
// 'SEK - Swedish Krona' => '',
+ // 'The project identifier is an optional alphanumeric code used to identify your project.' => '',
+ // 'Identifier' => '',
+ // 'Postmark (incoming emails)' => '',
+ // 'Help on Postmark integration' => '',
);
diff --git a/app/Locale/zh_CN/translations.php b/app/Locale/zh_CN/translations.php
index e6166335..63570ca2 100644
--- a/app/Locale/zh_CN/translations.php
+++ b/app/Locale/zh_CN/translations.php
@@ -858,4 +858,8 @@ return array(
// 'Take a screenshot and press CTRL+V or ⌘+V to paste here.' => '',
// 'Screenshot uploaded successfully.' => '',
// 'SEK - Swedish Krona' => '',
+ // 'The project identifier is an optional alphanumeric code used to identify your project.' => '',
+ // 'Identifier' => '',
+ // 'Postmark (incoming emails)' => '',
+ // 'Help on Postmark integration' => '',
);
diff --git a/app/Model/Project.php b/app/Model/Project.php
index dbb9db1b..231d57e7 100644
--- a/app/Model/Project.php
+++ b/app/Model/Project.php
@@ -60,6 +60,18 @@ class Project extends Base
}
/**
+ * Get a project by the identifier (code)
+ *
+ * @access public
+ * @param string $identifier
+ * @return array
+ */
+ public function getByIdentifier($identifier)
+ {
+ return $this->db->table(self::TABLE)->eq('identifier', strtoupper($identifier))->findOne();
+ }
+
+ /**
* Fetch project data by using the token
*
* @access public
@@ -276,6 +288,10 @@ class Project extends Base
$values['last_modified'] = time();
$values['is_private'] = empty($values['is_private']) ? 0 : 1;
+ if (! empty($values['identifier'])) {
+ $values['identifier'] = strtoupper($values['identifier']);
+ }
+
if (! $this->db->table(self::TABLE)->save($values)) {
$this->db->cancelTransaction();
return false;
@@ -338,6 +354,10 @@ class Project extends Base
*/
public function update(array $values)
{
+ if (! empty($values['identifier'])) {
+ $values['identifier'] = strtoupper($values['identifier']);
+ }
+
return $this->exists($values['id']) &&
$this->db->table(self::TABLE)->eq('id', $values['id'])->save($values);
}
@@ -443,7 +463,10 @@ class Project extends Base
new Validators\Integer('is_active', t('This value must be an integer')),
new Validators\Required('name', t('The project name is required')),
new Validators\MaxLength('name', t('The maximum length is %d characters', 50), 50),
+ new Validators\MaxLength('identifier', t('The maximum length is %d characters', 50), 50),
+ new Validators\AlphaNumeric('identifier', t('This value must be alphanumeric')) ,
new Validators\Unique('name', t('This project must be unique'), $this->db->getConnection(), self::TABLE),
+ new Validators\Unique('identifier', t('The identifier must be unique'), $this->db->getConnection(), self::TABLE),
);
}
@@ -456,6 +479,10 @@ class Project extends Base
*/
public function validateCreation(array $values)
{
+ if (! empty($values['identifier'])) {
+ $values['identifier'] = strtoupper($values['identifier']);
+ }
+
$v = new Validator($values, $this->commonValidationRules());
return array(
@@ -473,6 +500,10 @@ class Project extends Base
*/
public function validateModification(array $values)
{
+ if (! empty($values['identifier'])) {
+ $values['identifier'] = strtoupper($values['identifier']);
+ }
+
$rules = array(
new Validators\Required('id', t('This value is required')),
);
diff --git a/app/Model/User.php b/app/Model/User.php
index 6c348caa..d9f174bd 100644
--- a/app/Model/User.php
+++ b/app/Model/User.php
@@ -142,6 +142,18 @@ class User extends Base
}
/**
+ * Get a specific user by the email address
+ *
+ * @access public
+ * @param string $email Email
+ * @return array
+ */
+ public function getByEmail($email)
+ {
+ return $this->db->table(self::TABLE)->eq('email', $email)->findOne();
+ }
+
+ /**
* Get all users
*
* @access public
diff --git a/app/Schema/Mysql.php b/app/Schema/Mysql.php
index 3e67387d..22f8c1b0 100644
--- a/app/Schema/Mysql.php
+++ b/app/Schema/Mysql.php
@@ -6,7 +6,12 @@ use PDO;
use Core\Security;
use Model\Link;
-const VERSION = 65;
+const VERSION = 66;
+
+function version_66($pdo)
+{
+ $pdo->exec("ALTER TABLE projects ADD COLUMN identifier VARCHAR(50) DEFAULT ''");
+}
function version_65($pdo)
{
diff --git a/app/Schema/Postgres.php b/app/Schema/Postgres.php
index 6973b266..db30af67 100644
--- a/app/Schema/Postgres.php
+++ b/app/Schema/Postgres.php
@@ -6,7 +6,12 @@ use PDO;
use Core\Security;
use Model\Link;
-const VERSION = 46;
+const VERSION = 47;
+
+function version_47($pdo)
+{
+ $pdo->exec("ALTER TABLE projects ADD COLUMN identifier VARCHAR(50) DEFAULT ''");
+}
function version_46($pdo)
{
diff --git a/app/Schema/Sqlite.php b/app/Schema/Sqlite.php
index c4ebd98a..79c50458 100644
--- a/app/Schema/Sqlite.php
+++ b/app/Schema/Sqlite.php
@@ -6,7 +6,12 @@ use Core\Security;
use PDO;
use Model\Link;
-const VERSION = 64;
+const VERSION = 65;
+
+function version_65($pdo)
+{
+ $pdo->exec("ALTER TABLE projects ADD COLUMN identifier TEXT DEFAULT ''");
+}
function version_64($pdo)
{
diff --git a/app/ServiceProvider/ClassProvider.php b/app/ServiceProvider/ClassProvider.php
index b08d506c..0c02058e 100644
--- a/app/ServiceProvider/ClassProvider.php
+++ b/app/ServiceProvider/ClassProvider.php
@@ -78,6 +78,7 @@ class ClassProvider implements ServiceProviderInterface
'BitbucketWebhook',
'Hipchat',
'SlackWebhook',
+ 'PostmarkWebhook',
)
);
diff --git a/app/Template/config/integrations.php b/app/Template/config/integrations.php
index e11b62f8..b7dab2b4 100644
--- a/app/Template/config/integrations.php
+++ b/app/Template/config/integrations.php
@@ -6,7 +6,13 @@
<?= $this->formCsrf() ?>
- <h3><?= t('Gravatar') ?></h3>
+ <h3><img src="assets/img/postmark-icon.png"/>&nbsp;<?= t('Postmark (incoming emails)') ?></h3>
+ <div class="listing">
+ <input type="text" class="auto-select" readonly="readonly" value="<?= $this->getCurrentBaseUrl().$this->u('webhook', 'postmark', array('token' => $values['webhook_token'])) ?>"/><br/>
+ <p class="form-help"><a href="http://kanboard.net/documentation/postmark" target="_blank"><?= t('Help on Postmark integration') ?></a></p>
+ </div>
+
+ <h3><img src="assets/img/gravatar-icon.png"/>&nbsp;<?= t('Gravatar') ?></h3>
<div class="listing">
<?= $this->formCheckbox('integration_gravatar', t('Enable Gravatar images'), 1, $values['integration_gravatar'] == 1) ?>
</div>
diff --git a/app/Template/project/edit.php b/app/Template/project/edit.php
index c1f98315..ffd9be00 100644
--- a/app/Template/project/edit.php
+++ b/app/Template/project/edit.php
@@ -9,6 +9,10 @@
<?= $this->formLabel(t('Name'), 'name') ?>
<?= $this->formText('name', $values, $errors, array('required', 'maxlength="50"')) ?>
+ <?= $this->formLabel(t('Identifier'), 'identifier') ?>
+ <?= $this->formText('identifier', $values, $errors, array('maxlength="50"')) ?>
+ <p class="form-help"><?= t('The project identifier is an optional alphanumeric code used to identify your project.') ?></p>
+
<?= $this->formLabel(t('Description'), 'description') ?>
<div class="form-tabs">
diff --git a/app/Template/project/index.php b/app/Template/project/index.php
index 05a7d955..8d2bc30b 100644
--- a/app/Template/project/index.php
+++ b/app/Template/project/index.php
@@ -15,6 +15,7 @@
<tr>
<th class="column-8"><?= $paginator->order(t('Id'), 'id') ?></th>
<th class="column-8"><?= $paginator->order(t('Status'), 'is_active') ?></th>
+ <th class="column-8"><?= $paginator->order(t('Identifier'), 'identifier') ?></th>
<th class="column-20"><?= $paginator->order(t('Project'), 'name') ?></th>
<th><?= t('Columns') ?></th>
</tr>
@@ -31,6 +32,9 @@
<?php endif ?>
</td>
<td>
+ <?= $this->e($project['identifier']) ?>
+ </td>
+ <td>
<?= $this->a('<i class="fa fa-table"></i>', 'board', 'show', array('project_id' => $project['id']), false, 'dashboard-table-link', t('Board')) ?>&nbsp;
<?php if ($project['is_public']): ?>
diff --git a/assets/img/gravatar-icon.png b/assets/img/gravatar-icon.png
new file mode 100644
index 00000000..d9b3e651
--- /dev/null
+++ b/assets/img/gravatar-icon.png
Binary files differ
diff --git a/assets/img/postmark-icon.png b/assets/img/postmark-icon.png
new file mode 100644
index 00000000..87f07594
--- /dev/null
+++ b/assets/img/postmark-icon.png
Binary files differ
diff --git a/composer.json b/composer.json
index 79ec0abf..9e08dfc7 100644
--- a/composer.json
+++ b/composer.json
@@ -13,7 +13,8 @@
"symfony/console" : "@stable",
"symfony/event-dispatcher" : "~2.6",
"fguillot/simpleLogger" : "0.0.1",
- "christian-riesen/otp": "1.4"
+ "christian-riesen/otp": "1.4",
+ "nickcernis/html-to-markdown": "2.2.1"
},
"autoload" : {
"psr-0" : {
diff --git a/composer.lock b/composer.lock
index 22602c49..a50f9480 100644
--- a/composer.lock
+++ b/composer.lock
@@ -4,7 +4,7 @@
"Read more about it at http://getcomposer.org/doc/01-basic-usage.md#composer-lock-the-lock-file",
"This file is @generated automatically"
],
- "hash": "01ebe465ed3a59d8350670ebd4ef8793",
+ "hash": "1799891b06d5a8a516a48fefd429a3ed",
"packages": [
{
"name": "christian-riesen/base32",
@@ -357,6 +357,52 @@
"time": "2014-09-05 15:19:58"
},
{
+ "name": "nickcernis/html-to-markdown",
+ "version": "2.2.1",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/nickcernis/html-to-markdown.git",
+ "reference": "7263d2ce65011b050fa7ecda0cbe09b23e84271d"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/nickcernis/html-to-markdown/zipball/7263d2ce65011b050fa7ecda0cbe09b23e84271d",
+ "reference": "7263d2ce65011b050fa7ecda0cbe09b23e84271d",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=5.3"
+ },
+ "require-dev": {
+ "php": ">=5.3.3",
+ "phpunit/phpunit": "4.*"
+ },
+ "type": "library",
+ "autoload": {
+ "classmap": [
+ "HTML_To_Markdown.php"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Nick Cernis",
+ "email": "nick@cern.is",
+ "homepage": "http://modernnerd.net"
+ }
+ ],
+ "description": "An HTML-to-markdown conversion helper for PHP",
+ "homepage": "https://github.com/nickcernis/html-to-markdown",
+ "keywords": [
+ "html",
+ "markdown"
+ ],
+ "time": "2015-02-22 12:59:02"
+ },
+ {
"name": "pimple/pimple",
"version": "v3.0.0",
"source": {
diff --git a/docs/postmark.markdown b/docs/postmark.markdown
new file mode 100644
index 00000000..f0469dad
--- /dev/null
+++ b/docs/postmark.markdown
@@ -0,0 +1,59 @@
+Postmark
+========
+
+You can use the service [Postmark](https://postmarkapp.com/) to create tasks directly by email.
+
+This integration works with the inbound email service of Postmark.
+Kanboard use a webhook to handle incoming emails.
+
+Incoming emails workflow
+------------------------
+
+1. You send an email to a specific address, by example **something+myproject@inbound.mydomain.tld**
+2. Your email is forwarded to Postmark SMTP servers
+3. Postmark call the Kanboard webhook with the email in JSON format
+4. Kanboard parse the received email and create the task to the right project
+
+Note: New tasks are automatically created in the first column.
+
+Email format
+------------
+
+- The local part of the email address must use the plus separator, by example **kanboard+project123**
+- The string defined after the plus sign must match a project identifier, by example **project123** is the identifier of the project **Project 123**
+
+Email format
+------------
+
+- The email subject becomes the task subject
+- The email body becomes the task description (Markdown format)
+
+Incoming emails can be written in text or HTML formats.
+**Kanboard is able to convert simple HTML emails to Markdown**.
+
+Security and requirements
+-------------------------
+
+- The Kanboard webhook is protected by a random token
+- The sender email address (From header) must match a Kanboard user
+- The Kanboard project must have a unique identifier, by example **MYPROJECT**
+- The Kanboard user must be member of the project
+
+Postmark configuration
+----------------------
+
+- Follow the [official documentation about inbound email processing](http://developer.postmarkapp.com/developer-process-configure.html)
+- The Kanboard webhook url is displayed in **Settings > Integrations > Postmark**
+
+Kanboard configuration
+----------------------
+
+1. Be sure that your users have an email address in their profiles
+2. Assign a project identifier to the desired projects: **Project settings > Edit**
+3. Try to send an email to your project
+
+Troubleshootings
+----------------
+
+- Test the webhook url from the Postmark console, you should have a status code `200 OK`
+- Double-check requirements mentioned above
diff --git a/tests/units/Base.php b/tests/units/Base.php
index bce65f2d..b6302942 100644
--- a/tests/units/Base.php
+++ b/tests/units/Base.php
@@ -6,6 +6,8 @@ require __DIR__.'/../../app/constants.php';
use Symfony\Component\EventDispatcher\EventDispatcher;
use Symfony\Component\EventDispatcher\Debug\TraceableEventDispatcher;
use Symfony\Component\Stopwatch\Stopwatch;
+use SimpleLogger\Logger;
+use SimpleLogger\File;
date_default_timezone_set('UTC');
@@ -38,6 +40,9 @@ abstract class Base extends PHPUnit_Framework_TestCase
);
$this->container['db']->log_queries = true;
+
+ $this->container['logger'] = new Logger;
+ $this->container['logger']->setLogger(new File('/dev/null'));
}
public function tearDown()
diff --git a/tests/units/PostmarkWebhookTest.php b/tests/units/PostmarkWebhookTest.php
new file mode 100644
index 00000000..34be8515
--- /dev/null
+++ b/tests/units/PostmarkWebhookTest.php
@@ -0,0 +1,83 @@
+<?php
+
+require_once __DIR__.'/Base.php';
+
+use Integration\PostmarkWebhook;
+use Model\TaskCreation;
+use Model\TaskFinder;
+use Model\Project;
+use Model\ProjectPermission;
+use Model\User;
+
+class PostmarkWebhookTest extends Base
+{
+ public function testHandlePayload()
+ {
+ $w = new PostmarkWebhook($this->container);
+ $p = new Project($this->container);
+ $pp = new ProjectPermission($this->container);
+ $u = new User($this->container);
+ $tc = new TaskCreation($this->container);
+ $tf = new TaskFinder($this->container);
+
+ $this->assertEquals(2, $u->create(array('name' => 'me', 'email' => 'me@localhost')));
+
+ $this->assertEquals(1, $p->create(array('name' => 'test1')));
+ $this->assertEquals(2, $p->create(array('name' => 'test2', 'identifier' => 'TEST1')));
+
+ // Empty payload
+ $this->assertFalse($w->parsePayload(array()));
+
+ // Unknown user
+ $this->assertFalse($w->parsePayload(array('From' => 'a@b.c', 'Subject' => 'Email task', 'MailboxHash' => 'foobar', 'TextBody' => 'boo')));
+
+ // Project not found
+ $this->assertFalse($w->parsePayload(array('From' => 'me@localhost', 'Subject' => 'Email task', 'MailboxHash' => 'test', 'TextBody' => 'boo')));
+
+ // User is not member
+ $this->assertFalse($w->parsePayload(array('From' => 'me@localhost', 'Subject' => 'Email task', 'MailboxHash' => 'test1', 'TextBody' => 'boo')));
+ $this->assertTrue($pp->addMember(2, 2));
+
+ // The task must be created
+ $this->assertTrue($w->parsePayload(array('From' => 'me@localhost', 'Subject' => 'Email task', 'MailboxHash' => 'test1', 'TextBody' => 'boo')));
+
+ $task = $tf->getById(1);
+ $this->assertNotEmpty($task);
+ $this->assertEquals(2, $task['project_id']);
+ $this->assertEquals('Email task', $task['title']);
+ $this->assertEquals('boo', $task['description']);
+ $this->assertEquals(2, $task['creator_id']);
+ }
+
+ public function testHtml2Markdown()
+ {
+ $w = new PostmarkWebhook($this->container);
+ $p = new Project($this->container);
+ $pp = new ProjectPermission($this->container);
+ $u = new User($this->container);
+ $tc = new TaskCreation($this->container);
+ $tf = new TaskFinder($this->container);
+
+ $this->assertEquals(2, $u->create(array('name' => 'me', 'email' => 'me@localhost')));
+ $this->assertEquals(1, $p->create(array('name' => 'test2', 'identifier' => 'TEST1')));
+ $this->assertTrue($pp->addMember(1, 2));
+
+ $this->assertTrue($w->parsePayload(array('From' => 'me@localhost', 'Subject' => 'Email task', 'MailboxHash' => 'test1', 'TextBody' => 'boo', 'HtmlBody' => '<p><strong>boo</strong></p>')));
+
+ $task = $tf->getById(1);
+ $this->assertNotEmpty($task);
+ $this->assertEquals(1, $task['project_id']);
+ $this->assertEquals('Email task', $task['title']);
+ $this->assertEquals('**boo**', $task['description']);
+ $this->assertEquals(2, $task['creator_id']);
+
+ $this->assertTrue($w->parsePayload(array('From' => 'me@localhost', 'Subject' => 'Email task', 'MailboxHash' => 'test1', 'TextBody' => '**boo**', 'HtmlBody' => '')));
+
+ $task = $tf->getById(2);
+ $this->assertNotEmpty($task);
+ $this->assertEquals(1, $task['project_id']);
+ $this->assertEquals('Email task', $task['title']);
+ $this->assertEquals('**boo**', $task['description']);
+ $this->assertEquals(2, $task['creator_id']);
+ }
+}
diff --git a/tests/units/ProjectTest.php b/tests/units/ProjectTest.php
index 4864a3ae..231d403f 100644
--- a/tests/units/ProjectTest.php
+++ b/tests/units/ProjectTest.php
@@ -203,4 +203,57 @@ class ProjectTest extends Base
$this->assertFalse($p->disablePublicAccess(123));
}
+
+ public function testIdentifier()
+ {
+ $p = new Project($this->container);
+
+ // Creation
+ $this->assertEquals(1, $p->create(array('name' => 'UnitTest1', 'identifier' => 'test1')));
+ $this->assertEquals(2, $p->create(array('name' => 'UnitTest2')));
+
+ $project = $p->getById(1);
+ $this->assertNotEmpty($project);
+ $this->assertEquals('TEST1', $project['identifier']);
+
+ $project = $p->getById(2);
+ $this->assertNotEmpty($project);
+ $this->assertEquals('', $project['identifier']);
+
+ // Update
+ $this->assertTrue($p->update(array('id' => '2', 'identifier' => 'test2')));
+
+ $project = $p->getById(2);
+ $this->assertNotEmpty($project);
+ $this->assertEquals('TEST2', $project['identifier']);
+
+ $project = $p->getByIdentifier('test1');
+ $this->assertNotEmpty($project);
+ $this->assertEquals('TEST1', $project['identifier']);
+
+ // Validation rules
+ $r = $p->validateCreation(array('name' => 'test', 'identifier' => 'TEST1'));
+ $this->assertFalse($r[0]);
+
+ $r = $p->validateCreation(array('name' => 'test', 'identifier' => 'test1'));
+ $this->assertFalse($r[0]);
+
+ $r = $p->validateModification(array('id' => 1, 'name' => 'test', 'identifier' => 'TEST1'));
+ $this->assertTrue($r[0]);
+
+ $r = $p->validateModification(array('id' => 1, 'name' => 'test', 'identifier' => 'test3'));
+ $this->assertTrue($r[0]);
+
+ $r = $p->validateModification(array('id' => 1, 'name' => 'test', 'identifier' => ''));
+ $this->assertTrue($r[0]);
+
+ $r = $p->validateModification(array('id' => 1, 'name' => 'test', 'identifier' => 'TEST2'));
+ $this->assertFalse($r[0]);
+
+ $r = $p->validateCreation(array('name' => 'test', 'identifier' => 'a-b-c'));
+ $this->assertFalse($r[0]);
+
+ $r = $p->validateCreation(array('name' => 'test', 'identifier' => 'test 123'));
+ $this->assertFalse($r[0]);
+ }
}