Added attachments to campaigns

This commit is contained in:
Andris Reinman 2016-09-09 22:12:03 +03:00
parent 89715c56fc
commit bfc6983c93
6 changed files with 444 additions and 115 deletions

View file

@ -12,6 +12,7 @@ let feed = require('../feed');
let log = require('npmlog');
let mailer = require('../mailer');
let caches = require('../caches');
let humanize = require('humanize');
let allowedKeys = ['description', 'from', 'address', 'subject', 'template', 'source_url', 'list', 'segment', 'html', 'text', 'tracking_disabled'];
@ -362,6 +363,120 @@ module.exports.get = (id, withSegment, callback) => {
});
};
module.exports.getAttachments = (campaign, callback) => {
campaign = Number(campaign) || 0;
if (campaign < 1) {
return callback(new Error('Missing Campaign ID'));
}
db.getConnection((err, connection) => {
if (err) {
return callback(err);
}
connection.query('SELECT `id`, `filename`, `size`, `created` FROM `attachments` WHERE `campaign`=?', [campaign], (err, rows) => {
connection.release();
if (err) {
return callback(err);
}
if (!rows || !rows.length) {
return callback(null, []);
}
let attachments = rows.map((row, i) => {
row = tools.convertKeys(row);
row.index = i + 1;
row.size = humanize.filesize(Number(row.size) || 0);
return row;
});
return callback(null, attachments);
});
});
};
module.exports.addAttachment = (id, attachment, callback) => {
let size = attachment.content ? attachment.content.length : 0;
if (!size) {
return callback(new Error('Emtpy or too large attahcment'));
}
db.getConnection((err, connection) => {
if (err) {
return callback(err);
}
let keys = ['campaign', 'size'];
let values = [id, size];
Object.keys(attachment).forEach(key => {
let value;
if (Buffer.isBuffer(attachment[key])) {
value = attachment[key];
} else {
value = typeof attachment[key] === 'number' ? attachment[key] : (attachment[key] || '').toString().trim();
}
key = tools.toDbKey(key);
keys.push(key);
values.push(value);
});
let query = 'INSERT INTO `attachments` (`' + keys.join('`, `') + '`) VALUES (' + values.map(() => '?').join(',') + ')';
connection.query(query, values, (err, result) => {
connection.release();
if (err) {
return callback(err);
}
let attachmentId = result && result.insertId || false;
return callback(null, attachmentId);
});
});
};
module.exports.deleteAttachment = (id, attachment, callback) => {
db.getConnection((err, connection) => {
if (err) {
return callback(err);
}
let query = 'DELETE FROM `attachments` WHERE `id`=? AND `campaign`=? LIMIT 1';
connection.query(query, [attachment, id], (err, result) => {
connection.release();
if (err) {
return callback(err);
}
let deleted = result && result.affectedRows || false;
return callback(null, deleted);
});
});
};
module.exports.getAttachment = (id, attachment, callback) => {
db.getConnection((err, connection) => {
if (err) {
return callback(err);
}
let query = 'SELECT `filename`, `content_type`, `content` FROM `attachments` WHERE `id`=? AND `campaign`=? LIMIT 1';
connection.query(query, [attachment, id], (err, rows) => {
connection.release();
if (err) {
return callback(err);
}
if (!rows || !rows.length) {
return callback(null, false);
}
let attachment = tools.convertKeys(rows[0]);
return callback(null, attachment);
});
});
};
module.exports.getLinks = (id, linkId, callback) => {
if (!callback && typeof linkId === 'function') {
callback = linkId;

View file

@ -1,3 +1,3 @@
{
"schemaVersion": 18
"schemaVersion": 19
}

View file

@ -12,6 +12,11 @@ let tools = require('../lib/tools');
let striptags = require('striptags');
let passport = require('../lib/passport');
let htmlescape = require('escape-html');
let multer = require('multer');
let uploadStorage = multer.memoryStorage();
let uploads = multer({
storage: uploadStorage
});
router.all('/*', (req, res, next) => {
if (!req.user) {
@ -120,6 +125,12 @@ router.get('/edit/:id', passport.csrfProtection, (req, res, next) => {
return res.redirect('/campaigns');
}
campaigns.getAttachments(campaign.id, (err, attachments) => {
if (err) {
return next(err);
}
campaign.attachments = attachments;
settings.list(['disableWysiwyg'], (err, configItems) => {
if (err) {
return next(err);
@ -151,6 +162,7 @@ router.get('/edit/:id', passport.csrfProtection, (req, res, next) => {
campaign.disableWysiwyg = configItems.disableWysiwyg;
campaign.showGeneral = req.query.tab === 'general' || !req.query.tab;
campaign.showTemplate = req.query.tab === 'template';
campaign.showAttachments = req.query.tab === 'attachments';
let view;
switch (campaign.type) {
@ -239,6 +251,7 @@ router.get('/edit/:id', passport.csrfProtection, (req, res, next) => {
});
});
});
});
});
router.post('/edit', passport.parseForm, passport.csrfProtection, (req, res) => {
@ -762,4 +775,80 @@ router.post('/inactivate', passport.parseForm, passport.csrfProtection, (req, re
});
});
router.post('/attachment', uploads.single('attachment'), passport.parseForm, passport.csrfProtection, (req, res) => {
campaigns.get(req.body.id, false, (err, campaign) => {
if (err || !campaign) {
req.flash('danger', err && err.message || err || 'Could not find campaign with specified ID');
return res.redirect('/campaigns');
}
campaigns.addAttachment(campaign.id, {
filename: req.file.originalname,
contentType: req.file.mimetype,
content: req.file.buffer
}, (err, attachmentId) => {
if (err) {
req.flash('danger', err && err.message || err);
} else if (attachmentId) {
req.flash('success', 'Attachment uploaded');
} else {
req.flash('info', 'Could not store attachment');
}
return res.redirect('/campaigns/edit/' + campaign.id + '?tab=attachments');
});
});
});
router.post('/attachment/delete', passport.parseForm, passport.csrfProtection, (req, res) => {
campaigns.get(req.body.id, false, (err, campaign) => {
if (err || !campaign) {
req.flash('danger', err && err.message || err || 'Could not find campaign with specified ID');
return res.redirect('/campaigns');
}
campaigns.deleteAttachment(campaign.id, Number(req.body.attachment), (err, deleted) => {
if (err) {
req.flash('danger', err && err.message || err);
} else if (deleted) {
req.flash('success', 'Attachment deleted');
} else {
req.flash('info', 'Could not delete attachment');
}
return res.redirect('/campaigns/edit/' + campaign.id + '?tab=attachments');
});
});
});
router.post('/attachment/download', passport.parseForm, passport.csrfProtection, (req, res) => {
campaigns.get(req.body.id, false, (err, campaign) => {
if (err || !campaign) {
req.flash('danger', err && err.message || err || 'Could not find campaign with specified ID');
return res.redirect('/campaigns');
}
campaigns.getAttachment(campaign.id, Number(req.body.attachment), (err, attachment) => {
if (err) {
req.flash('danger', err && err.message || err);
return res.redirect('/campaigns/edit/' + campaign.id + '?tab=attachments');
} else if (!attachment) {
req.flash('success', 'Attachment uploaded');
return res.redirect('/campaigns/edit/' + campaign.id + '?tab=attachments');
}
res.set('Content-Disposition', 'attachment; filename="' + encodeURIComponent(attachment.filename).replace(/['()]/g, escape) + '"');
res.set('Content-Type', attachment.contentType);
res.send(attachment.content);
});
});
});
router.get('/attachment/:campaign', passport.csrfProtection, (req, res) => {
campaigns.get(req.params.campaign, false, (err, campaign) => {
if (err || !campaign) {
req.flash('danger', err && err.message || err || 'Could not find campaign with specified ID');
return res.redirect('/campaigns');
}
campaign.csrfToken = req.csrfToken();
res.render('campaigns/upload-attachment', campaign);
});
});
module.exports = router;

View file

@ -0,0 +1,21 @@
# Header section
# Define incrementing schema version number
SET @schema_version = '19';
CREATE TABLE `attachments` (
`id` int(11) unsigned NOT NULL AUTO_INCREMENT,
`campaign` int(11) unsigned NOT NULL,
`filename` varchar(255) CHARACTER SET utf8mb4 NOT NULL DEFAULT '',
`content_type` varchar(100) CHARACTER SET ascii NOT NULL DEFAULT '',
`content` longblob,
`size` int(11) NOT NULL,
`created` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP,
PRIMARY KEY (`id`),
KEY `campaign` (`campaign`),
CONSTRAINT `attachments_ibfk_1` FOREIGN KEY (`campaign`) REFERENCES `campaigns` (`id`) ON DELETE CASCADE
) ENGINE=InnoDB DEFAULT CHARSET=utf8;
# Footer section
LOCK TABLES `settings` WRITE;
INSERT INTO `settings` (`key`, `value`) VALUES('db_schema_version', @schema_version) ON DUPLICATE KEY UPDATE `value`=@schema_version;
UNLOCK TABLES;

View file

@ -17,16 +17,29 @@
<input type="hidden" name="id" value="{{id}}" />
</form>
{{#each attachments}}
<form method="post" id="attachment-download-{{id}}" action="/campaigns/attachment/download">
<input type="hidden" name="_csrf" value="{{../csrfToken}}">
<input type="hidden" name="id" value="{{../id}}" />
<input type="hidden" name="attachment" value="{{id}}" />
</form>
<form method="post" class="delete-form" id="attachment-delete-{{id}}" action="/campaigns/attachment/delete">
<input type="hidden" name="_csrf" value="{{../csrfToken}}">
<input type="hidden" name="id" value="{{../id}}" />
<input type="hidden" name="attachment" value="{{id}}" />
</form>
{{/each}}
<form class="form-horizontal" method="post" action="/campaigns/edit">
<input type="hidden" name="_csrf" value="{{csrfToken}}">
<input type="hidden" name="id" value="{{id}}" />
<div>
<!-- Nav tabs -->
<ul class="nav nav-tabs" role="tablist">
<li role="presentation" class="{{#if showGeneral}}active{{/if}}"><a href="#general" aria-controls="general" role="tab" data-toggle="tab">General</a></li>
<li role="presentation" class="{{#if showTemplate}}active{{/if}}"><a href="#template" aria-controls="template" role="tab" data-toggle="tab">Template</a></li>
<li role="presentation" class="{{#if showAttachments}}active{{/if}}"><a href="#attachments" aria-controls="attachments" role="tab" data-toggle="tab">Attachments{{#if attachments}} <span class="badge">{{attachments.length}}</span>{{/if}}</a></li>
</ul>
<div class="tab-content">
@ -109,7 +122,6 @@
</fieldset>
</div>
<div role="tabpanel" class="tab-pane {{#if showTemplate}}active{{/if}}" id="template">
<p></p>
<fieldset>
@ -187,10 +199,70 @@
{{/if}}
</fieldset>
</div>
<div role="tabpanel" class="tab-pane {{#if showAttachments}}active{{/if}}" id="attachments">
<p></p>
<fieldset>
<legend>
Attachments
</legend>
<div class="table-responsive">
<table class="table table-bordered table-hover">
<thead>
<th class="col-md-1">
#
</th>
<th>
File
</th>
<th class="col-md-1">
Size
</th>
<th class="col-md-1">
&nbsp;
</th>
</thead>
<tbody>
{{#if attachments}}
{{#each attachments}}
<tr>
<td>
{{index}}
</td>
<td>
<button type="submit" form="attachment-download-{{id}}" class="btn btn-link btn-xs"><i class="glyphicon glyphicon-cloud-download"></i> {{filename}}</button>
</td>
<td>
{{size}}
</td>
<td>
<button type="submit" form="attachment-delete-{{id}}" class="btn btn-danger btn-xs"><i class="glyphicon glyphicon-remove"></i> Delete</button>
</td>
</tr>
{{/each}}
{{else}}
<tr>
<td colspan="4">
No data available in table
</td>
</tr>
{{/if}}
</tbody>
</table>
</div>
<div class="pull-right">
<a class="btn btn-info btn-sm" href="/campaigns/attachment/{{id}}" role="button"><span class="glyphicon glyphicon-plus" aria-hidden="true"></span> Add Attachment</a>
</div>
</fieldset>
</div>
<hr />
</div>
<hr/>
<div class="form-group">
<div class="col-sm-offset-2 col-sm-10">
@ -200,4 +272,5 @@
<button type="submit" class="btn btn-primary"><i class="glyphicon glyphicon-ok"></i> Update</button>
</div>
</div>
</div>
</form>

View file

@ -0,0 +1,31 @@
<ol class="breadcrumb">
<li><a href="/">Home</a></li>
<li><a href="/campaigns">Campaigns</a></li>
{{#if parent}}
<li><a href="/campaigns/view/{{parent.id}}">{{parent.name}}</a></li>
{{/if}}
<li><a href="/campaigns/view/{{id}}">{{name}}</a></li>
<li><a href="/campaigns/edit/{{id}}?tab=attachments">Edit Campaign</a></li>
<li class="active">Add Attachment</li>
</ol>
<h2>Edit Campaign <a class="btn btn-default btn-xs" href="/campaigns/view/{{id}}" role="button"><span class="glyphicon glyphicon-arrow-left" aria-hidden="true"></span> View campaign</a></h2>
<hr>
<form class="form-horizontal" method="post" action="/campaigns/attachment" enctype="multipart/form-data">
<input type="hidden" name="_csrf" value="{{csrfToken}}">
<input type="hidden" name="id" value="{{id}}" />
<div class="form-group">
<label for="attachment" class="col-sm-2 control-label">Upload new</label>
<div class="col-sm-6">
<input type="file" class="form-control" name="attachment" id="attachment" required>
</div>
</div>
<div class="form-group">
<div class="col-sm-offset-2 col-sm-10">
<button type="submit" class="btn btn-primary"><i class="glyphicon glyphicon-cloud-upload"></i> Upload</button>
</div>
</div>
</form>