diff options
| author | Dan Goodliffe <dan@randomdan.homeip.net> | 2016-12-02 21:27:50 +0000 | 
|---|---|---|
| committer | Dan Goodliffe <dan@randomdan.homeip.net> | 2016-12-02 21:27:50 +0000 | 
| commit | 9101b953e4434d1a50aa42d7329dfb8223a0b333 (patch) | |
| tree | e561e24bcb81407bc1cc1fd8b729dcfa9858b81c | |
| parent | Fix use flag ordering to be not locale sensitive and resulting changes in tes... (diff) | |
| download | gentoobrowse-api-9101b953e4434d1a50aa42d7329dfb8223a0b333.tar.bz2 gentoobrowse-api-9101b953e4434d1a50aa42d7329dfb8223a0b333.tar.xz gentoobrowse-api-9101b953e4434d1a50aa42d7329dfb8223a0b333.zip | |
Basic notification service implementation
17 files changed, 645 insertions, 1 deletions
| diff --git a/gentoobrowse-api/service/Jamfile.jam b/gentoobrowse-api/service/Jamfile.jam index a065393..7300b68 100644 --- a/gentoobrowse-api/service/Jamfile.jam +++ b/gentoobrowse-api/service/Jamfile.jam @@ -4,10 +4,11 @@ lib icetray : : : : <include>/usr/include/icetray ;  lib git2 ;  lib gentoobrowse-service : -	[ glob-tree *.cpp : bin ] +	[ glob-tree *.cpp : bin notifications ]  	[ glob-tree *.sql ]  	:  	<library>..//adhocutil +	<library>notifications  	<library>icetray  	<library>git2  	<library>..//dbppcore @@ -15,6 +16,8 @@ lib gentoobrowse-service :  	<library>..//slicer  	<library>..//slicer-db  	<library>../api//gentoobrowse-api +	<library>notifications//gentoobrowse-service-notifications +	<implicit-dependency>notifications//gentoobrowse-service-notifications  	<implicit-dependency>../api//gentoobrowse-api  	<library>..//boost_system  	<library>..//boost_thread @@ -27,6 +30,8 @@ lib gentoobrowse-service :  	: :  	<include>.  	<implicit-dependency>../api//gentoobrowse-api +	<library>notifications//gentoobrowse-service-notifications +	<implicit-dependency>notifications//gentoobrowse-service-notifications  	<library>../api//gentoobrowse-api  	; diff --git a/gentoobrowse-api/service/main.cpp b/gentoobrowse-api/service/main.cpp index bdf6ed5..b7b18a2 100644 --- a/gentoobrowse-api/service/main.cpp +++ b/gentoobrowse-api/service/main.cpp @@ -4,6 +4,8 @@  #include "portageimpl.h"  #include "usersimpl.h"  #include "maintenanceimpl.h" +#include "notificationsimpl.h" +#include "mailserverimpl.h"  namespace Gentoo {  namespace Service { @@ -15,6 +17,8 @@ class Api : public IceTray::Service {  			adp->add(new Portage(db), ic->stringToIdentity("portage"));  			adp->add(new Maintenance(db), ic->stringToIdentity("maintenance"));  			adp->add(new Users(db), ic->stringToIdentity("users")); +			adp->add(new Notifications(), ic->stringToIdentity("notifications")); +			adp->add(new MailServer(), ic->stringToIdentity("mailserver"));  		}  }; diff --git a/gentoobrowse-api/service/notifications/Jamfile.jam b/gentoobrowse-api/service/notifications/Jamfile.jam new file mode 100644 index 0000000..a168649 --- /dev/null +++ b/gentoobrowse-api/service/notifications/Jamfile.jam @@ -0,0 +1,44 @@ +import type : register ; +import generators : register-standard ; + +type.register XSLT : xslt ; + +generators.register-standard xslt.h : XSLT : H ; + +h base-xslt : xslt/base.xslt : <slicer>yes ; +h signup-xslt : xslt/signup.xslt : <slicer>yes ; +h news-xslt : xslt/news.xslt : <slicer>yes ; + +lib exslt ; +lib xslt : : : : <include>/usr/include/libxslt ; +lib xml2 : : : : <include>/usr/include/libxml2 ; +lib esmtp ; +lib slicer-xml ; +lib boost_system ; + +lib gentoobrowse-service-notifications : +	[ glob *.ice *.cpp ] +	: +	<slicer>yes +	<library>xslt +	<library>xml2 +	<library>esmtp +	<library>../../..//libxmlpp +	<library>boost_system +	<library>slicer-xml +	<library>../../domain//gentoobrowse-domain +	<implicit-dependency>../../domain//gentoobrowse-domain +	<dependency>base-xslt +	<dependency>signup-xslt +	<dependency>news-xslt +	: : +	<include>. +	; + +actions xslt.h +{ +	( cd $(2:D) ; xxd -i $(2:B)$(2:S) ) > $(1) +} + +IMPORT $(__name__) : xslt.h : : xslt.h ; + diff --git a/gentoobrowse-api/service/notifications/mailserverimpl.cpp b/gentoobrowse-api/service/notifications/mailserverimpl.cpp new file mode 100644 index 0000000..c3dfdfa --- /dev/null +++ b/gentoobrowse-api/service/notifications/mailserverimpl.cpp @@ -0,0 +1,64 @@ +#include "mailserverimpl.h" +#include <libesmtp.h> +#include <Ice/ObjectAdapter.h> +#include <Ice/Communicator.h> + +namespace Gentoo { +	namespace Service { +		typedef std::vector<std::string> Parts; +		typedef std::pair<Parts::iterator, Parts::const_iterator> PartsProgress; + +		void +		MailServer::sendEmail(const Gentoo::EmailPtr & msg, const Ice::Current & c) +		{ +			auto props = c.adapter->getCommunicator()->getProperties(); +			smtp_session_t session = smtp_create_session(); +			smtp_message_t message = smtp_add_message(session); +			auto server = props->getPropertyWithDefault("GentooBrowseAPI.MailServer", "localhost"); +			smtp_set_server(session, server.c_str()); +			smtp_set_header(message, "To", msg->to.name.c_str(), msg->to.address.c_str()); +			smtp_set_header(message, "From", msg->from.name.c_str(), msg->from.address.c_str()); +			smtp_set_header(message, "Subject", msg->subject.c_str()); +			Parts parts; +			parts.emplace_back("Content-Type: multipart/alternative; boundary=\"<<divider>>\"\r\n"); +			parts.emplace_back("MIME-Version: 1.0\r\n"); +			parts.emplace_back("Content-Transfer-Encoding: binary\r\n"); +			smtp_add_recipient(message, msg->to.address.c_str()); +			for (const auto & p : msg->body) { +				parts.emplace_back("\r\n--<<divider>>\r\nContent-Type: " + p->mimetype  + "; utf-8\r\n\r\n"); +				parts.emplace_back(p->payload); +			} +			parts.emplace_back("\r\n--<<divider>>--\r\n"); +			PartsProgress pp { parts.begin(), parts.end() }; +			smtp_set_messagecb(message, writeBody, &pp); +			if (!smtp_start_session(session)) { +				char buf[BUFSIZ]; +				auto b = smtp_strerror(smtp_errno(), buf, sizeof(buf)); +				assert(b); +				SendEmailFailed e(b); +				smtp_destroy_session(session); +				throw e; +			} +			smtp_destroy_session(session); +		} + +		const char * +		MailServer::writeBody(void **, int * len, void * arg) +		{ +			auto parts = static_cast<PartsProgress *>(arg); +			if (len == NULL || parts->first == parts->second) { +				return NULL; +			} +			const auto & p = *parts->first++; +			*len = p.length(); +			return p.data(); +		} +	} + +	void +	SendEmailFailed::ice_print(std::ostream & buf) const +	{ +		buf << "Failed to send email: " << message; +	} +} + diff --git a/gentoobrowse-api/service/notifications/mailserverimpl.h b/gentoobrowse-api/service/notifications/mailserverimpl.h new file mode 100644 index 0000000..23b17c2 --- /dev/null +++ b/gentoobrowse-api/service/notifications/mailserverimpl.h @@ -0,0 +1,20 @@ +#ifndef MAILSERVERIMPL_H +#define MAILSERVERIMPL_H + +#include <notifications.h> +#include <visibility.h> + +namespace Gentoo { +	namespace Service { +		class DLL_PUBLIC MailServer : public Gentoo::MailServer { +			public: +				void sendEmail(const Gentoo::EmailPtr & msg, const Ice::Current &) override; + +			private: +				static const char * writeBody(void ** buf, int * len, void * arg); +		}; +	} +} + +#endif + diff --git a/gentoobrowse-api/service/notifications/notifications.ice b/gentoobrowse-api/service/notifications/notifications.ice new file mode 100644 index 0000000..a60fa28 --- /dev/null +++ b/gentoobrowse-api/service/notifications/notifications.ice @@ -0,0 +1,49 @@ +#ifndef GENTOO_NOTIFICATIONS +#define GENTOO_NOTIFICATIONS + +#include <user-models.ice> +#include <portage-models.ice> + +module Gentoo { +	class MimePart { +		string mimetype; +		string payload; +	}; + +	sequence<MimePart> MimeParts; + +	struct Address { +		string name; +		string address; +	}; + +	class Email { +		Address to; +		Address from; +		string subject; +		MimeParts body; +	}; + +	["cpp:ice_print"] +	exception SendEmailFailed { +		string message; +	}; + +	struct NewsContent { +		Categories categories; +		Packages packages; +		Ebuilds ebuilds; +	}; + +	interface Notifications { +		Email getSignup(NewUser user); +		Email getNews(User user, NewsContent nc); +	}; + +	interface MailServer { +		idempotent void sendEmail(Email msg) throws SendEmailFailed; +	}; +}; + +#endif + diff --git a/gentoobrowse-api/service/notifications/notificationsimpl.cpp b/gentoobrowse-api/service/notifications/notificationsimpl.cpp new file mode 100644 index 0000000..d7dbee4 --- /dev/null +++ b/gentoobrowse-api/service/notifications/notificationsimpl.cpp @@ -0,0 +1,59 @@ +#include "notificationsimpl.h" +#include <libxml/parser.h> +#include <base-xslt.h> +#include <news-xslt.h> +#include <signup-xslt.h> +#include <string.h> +#include <slicer/slicer.h> +#include "xsltStreamSerializer.h" + +namespace Gentoo { +	namespace Service { +		xmlDocPtr	xsltDocLoaderFunc	(const xmlChar * URI, xmlDictPtr, int, void *, xsltLoadType) +		{ +#define MATCH(name) \ +			if (xmlStrcmp(URI, BAD_CAST #name ".xslt") == 0) { \ +				return xmlParseMemory((char *)name ## _xslt, name ## _xslt_len); \ +			} +			MATCH(base) +			MATCH(news) +			MATCH(signup) +			return NULL; +#undef MATCH +		} + +		Notifications::Notifications() +		{ +			xsltSetLoaderFunc(&xsltDocLoaderFunc); +			news = xsltSSPtr(xsltParseStylesheetFile(BAD_CAST "news.xslt"), xsltFreeStylesheet); +			signup = xsltSSPtr(xsltParseStylesheetFile(BAD_CAST "signup.xslt"), xsltFreeStylesheet); +			xsltSetLoaderFunc(NULL); +		} + +		Gentoo::EmailPtr Notifications::getSignup(const Gentoo::NewUserPtr & u, const Ice::Current &) +		{ +			auto e = basicMail("Welcome", u); +			Slicer::SerializeAny<XsltStreamSerializer>(u, e, signup.get()); +			return e; +		} + +		Gentoo::EmailPtr Notifications::getNews(const Gentoo::UserPtr & u, const Gentoo::NewsContent & c, const Ice::Current &) +		{ +			auto e = basicMail("Latest updates", u); +			Slicer::SerializeAny<XsltStreamSerializer>(c, e, news.get()); +			return e; +		} + +		Gentoo::EmailPtr Notifications::basicMail(const std::string & subject, Gentoo::UserPtr u) +		{ +			Gentoo::EmailPtr e = new Gentoo::Email(); +			e->subject = "Gentoo Browse: " + subject; +			e->from.name = "Gentoo Browse"; +			e->from.address = "noreply@gentoobrowse.randomdan.homeip.net"; +			e->to.name = u->userrealname; +			e->to.address = u->useremail; +			return e; +		} +	} +} + diff --git a/gentoobrowse-api/service/notifications/notificationsimpl.h b/gentoobrowse-api/service/notifications/notificationsimpl.h new file mode 100644 index 0000000..1d3b1fc --- /dev/null +++ b/gentoobrowse-api/service/notifications/notificationsimpl.h @@ -0,0 +1,29 @@ +#ifndef NOTIFICATIONSIMPL_H +#define NOTIFICATIONSIMPL_H + +#include <notifications.h> +#include <visibility.h> +#include <memory> +#include <libxslt/documents.h> + +namespace Gentoo { +	namespace Service { +		class DLL_PUBLIC Notifications : public Gentoo::Notifications { +			public: +				Notifications(); + +				Gentoo::EmailPtr getSignup(const Gentoo::NewUserPtr &, const Ice::Current &) override; +				Gentoo::EmailPtr getNews(const Gentoo::UserPtr &, const Gentoo::NewsContent &, const Ice::Current &) override; + +			private: +				Gentoo::EmailPtr basicMail(const std::string &, Gentoo::UserPtr u); + +				typedef std::shared_ptr<xsltStylesheet> xsltSSPtr; +				xsltSSPtr news; +				xsltSSPtr signup; +		}; +	} +} + +#endif + diff --git a/gentoobrowse-api/service/notifications/xslt/base.xslt b/gentoobrowse-api/service/notifications/xslt/base.xslt new file mode 100644 index 0000000..9db7752 --- /dev/null +++ b/gentoobrowse-api/service/notifications/xslt/base.xslt @@ -0,0 +1,61 @@ +<?xml version="1.0" encoding="UTF-8" ?> +<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform" +		xmlns:project2="http://project2.randomdan.homeip.net" exclude-result-prefixes="project2" > +  <xsl:output encoding="utf-8" method="html" media-type="text/html" indent="yes" /> + +  <xsl:template match="/*"> +		<xsl:text disable-output-escaping="yes"><!DOCTYPE html>
</xsl:text> +		<html lang="en"> +			<head> +				<meta charset="utf-8" /> +				<link href="http://gentoobrowse.randomdan.homeip.net/css/bootstrap.min.css" rel="stylesheet" media="screen" /> +				<link href="http://gentoobrowse.randomdan.homeip.net/css/bootstrap-responsive.min.css" rel="stylesheet" media="screen" /> +				<link href="http://gentoobrowse.randomdan.homeip.net/css/general.css" rel="stylesheet" media="screen" /> +				<title><xsl:call-template name="title" /></title> +			</head> +			<body data-spy="scroll" data-target=".bs-docs-sidebar"> +				<!-- nav --> +				<nav class="navbar navbar-inverse navbar-fixed-top"> +					<div class="navbar-inner"> +						<div class="container"> +							<a class="brand" href="http://gentoobrowse.randomdan.homeip.net/">Gentoo Browse</a> +						</div> +					</div> +				</nav> + +				<!-- page content --> +				<header class="jumbotron subhead"> +					<div class="container"> +						<h1> +							<xsl:call-template name="title" /> +						</h1> +					</div> +				</header> +				<div class="container"> +					<article class="row"> +						<xsl:call-template name="content" /> +						<section class="signoff"> +						</section> +					</article> +				</div> + +				<!-- footer --> +				<footer class="footer"> +					<div class="container"> +						<p>Gentoo Browse is not an official Gentoo website. The name "Gentoo" and the "g" logo are trademarks of the Gentoo Foundation, Inc.</p> +					</div> +				</footer> + +				<script src="http://code.jquery.com/jquery-latest.js"></script> +				<script src="http://gentoobrowse.randomdan.homeip.net/js/bootstrap.min.js"></script> +				<script src="http://gentoobrowse.randomdan.homeip.net/js/js.js"></script> +			</body> +		</html> +  </xsl:template> + +  <xsl:template name="head"> +  </xsl:template> + +  <xsl:template name="header"> +  </xsl:template> +</xsl:stylesheet> diff --git a/gentoobrowse-api/service/notifications/xslt/news.xslt b/gentoobrowse-api/service/notifications/xslt/news.xslt new file mode 100644 index 0000000..6b2ea29 --- /dev/null +++ b/gentoobrowse-api/service/notifications/xslt/news.xslt @@ -0,0 +1,57 @@ +<?xml version="1.0" encoding="UTF-8"?> +<xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform" xmlns:fn="http://exslt.org/strings" xmlns:project2="http://project2.randomdan.homeip.net" version="1.0" exclude-result-prefixes="project2 fn"> +  <xsl:import href="base.xslt"/> +  <xsl:output encoding="utf-8" method="html" media-type="text/html" indent="yes"/> + +  <xsl:template name="title">Latest news</xsl:template> + +	<xsl:template name="content"> +		<xsl:variable name="packages" select="packages/package[packageid = ../../ebuilds/ebuild/packageid]"/> +		<xsl:variable name="categories" select="categories/category[categoryid = $packages/categoryid]"/> +		<section> +			<h2 class="page-header">New versions of tracked packages</h2> +			<div class="span3 bs-docs-sidebar"> +				<ul class="nav nav-list bs-docs-sidenav"> +					<xsl:for-each select="$categories"> +						<xsl:sort select="name" data-type="text" order="ascending"/> +						<li> +							<a> +								<xsl:attribute name="href">http://gentoobrowse.randomdan.homeip.net/packages/<xsl:value-of select="name" /></xsl:attribute> +								<i class="icon-chevron-right"></i> +								<xsl:value-of select="name" /> +							</a> +						</li> +					</xsl:for-each> +				</ul> +			</div> +			<div class="span9"> +				<xsl:for-each select="$categories"> +					<xsl:sort select="name" data-type="text" order="ascending"/> +					<xsl:variable name="category" select="." /> +					<xsl:for-each select="$packages[categoryid = current()/categoryid]"> +						<xsl:sort select="name" data-type="text" order="ascending"/> +						<xsl:variable name="package" select="." /> +						<p> +							<a> +								<xsl:attribute name="href">http://gentoobrowse.randomdan.homeip.net/packages/<xsl:value-of select="$category/name" /></xsl:attribute> +								<xsl:value-of select="$category/name" /> +							</a> +							/ +							<a> +								<xsl:attribute name="href">http://gentoobrowse.randomdan.homeip.net/packages/<xsl:value-of select="$category/name" />/<xsl:value-of select="name" /></xsl:attribute> +								<xsl:value-of select="name" /> +							</a> +							: +							<xsl:value-of select="description" /> +						</p> +						<xsl:for-each select="../../ebuilds/ebuild[packageid = current()/packageid]"> +							<p> +								v<xsl:value-of select="version" /> +							</p> +						</xsl:for-each> +					</xsl:for-each> +				</xsl:for-each> +			</div> +		</section> +	</xsl:template> +</xsl:stylesheet> diff --git a/gentoobrowse-api/service/notifications/xslt/signup.xslt b/gentoobrowse-api/service/notifications/xslt/signup.xslt new file mode 100644 index 0000000..e579835 --- /dev/null +++ b/gentoobrowse-api/service/notifications/xslt/signup.xslt @@ -0,0 +1,24 @@ +<?xml version="1.0" encoding="UTF-8" ?> +<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform"> +	<xsl:import href="base.xslt"/> + +  <xsl:template name="title">Welcome to Gentoo Browse</xsl:template> + +  <xsl:template name="content"> +		<section> +			<h2 class="page-header">Confirmation</h2> +			<p> +				Click <a> +					<xsl:attribute name="href"> +						<xsl:text>http://gentoobrowse.randomdan.homeip.net/user/verify/</xsl:text> +						<xsl:value-of select="verifyguid" /> +					</xsl:attribute> +					<xsl:text>http://gentoobrowse.randomdan.homeip.net/user/verify/</xsl:text> +					<xsl:value-of select="verifyguid" /> +				</a> to confirm your account. +			</p> +			<p>Or alternatively, copy and paste this code into the confirmation page: <xsl:value-of select="verifyguid" /></p> +		</section> +  </xsl:template> +</xsl:stylesheet> + diff --git a/gentoobrowse-api/service/notifications/xsltStreamSerializer.cpp b/gentoobrowse-api/service/notifications/xsltStreamSerializer.cpp new file mode 100644 index 0000000..5a71b35 --- /dev/null +++ b/gentoobrowse-api/service/notifications/xsltStreamSerializer.cpp @@ -0,0 +1,82 @@ +#include "xsltStreamSerializer.h" +#include <libxslt/xsltInternals.h> +#include <libxml/HTMLtree.h> +#include <factory.impl.h> +#include <processPipes.h> +#include <sys/wait.h> + +namespace Gentoo { +	static int xmlstrmclosecallback(void * context) +	{ +		((std::ostream*)context)->flush(); +		return 0; +	} + +	static int xmlstrmwritecallback(void * context, const char * buffer, int len) +	{ +		((std::ostream*)context)->write(buffer, len); +		return len; +	} + +	XsltStreamSerializer::XsltStreamSerializer(Gentoo::EmailPtr e, xsltStylesheet * ss) : +		Slicer::XmlDocumentSerializer(doc), +		mail(e), +		doc(nullptr), +		stylesheet(ss) +	{ +	} + +	XsltStreamSerializer::~XsltStreamSerializer() +	{ +		delete doc; +	} + +	void +	XsltStreamSerializer::Serialize(Slicer::ModelPartForRootPtr mp) +	{ +		Slicer::XmlDocumentSerializer::Serialize(mp); +		auto result = std::shared_ptr<xmlDoc>(xsltApplyStylesheet(stylesheet, doc->cobj(), nullptr), xmlFreeDoc); +		if (!result) { +			throw xmlpp::exception("Failed to apply XSL transform"); +		} +		appendText(result.get()); +		appendHtml(result.get()); +	} + +	void XsltStreamSerializer::appendHtml(xmlDoc * result) const +	{ +		std::stringstream strm; +		xmlOutputBufferPtr buf = xmlOutputBufferCreateIO(xmlstrmwritecallback, xmlstrmclosecallback, &strm, NULL); +		htmlDocContentDumpFormatOutput(buf, result, "utf-8", 0); +		xmlOutputBufferClose(buf); +		mail->body.push_back(new Gentoo::MimePart("text/html", strm.str())); +	} + +	void XsltStreamSerializer::appendText(xmlDoc * result) const +	{ +		std::stringstream strm; +		std::vector<std::string> callLynx; +		callLynx.push_back("/usr/bin/lynx"); +		callLynx.push_back("-dump"); +		callLynx.push_back("-stdin"); +		std::string widthArg = "-width=105"; +		callLynx.push_back(widthArg); +		AdHoc::System::ProcessPipes fds(callLynx, true, true, false); +		FILE * lynxIn = fdopen(fds.fdIn(), "w"); +		// Fixed encoding as we want the result to go back into a ustring +		htmlNodeDumpFileFormat(lynxIn, result, xmlDocGetRootElement(result), "utf-8", 0); +		fclose(lynxIn); +		char buf[1024]; +		int r; +		while ((r = read(fds.fdOut(), buf, sizeof(buf))) > 0) { +			strm.write(buf, r); +		} +		int status; +		waitpid(fds.pid(), &status, 0); +		if (status != 0) { +			throw std::runtime_error("Lynx failed"); +		} +		mail->body.push_back(new Gentoo::MimePart("text/plain", strm.str())); +	} +} + diff --git a/gentoobrowse-api/service/notifications/xsltStreamSerializer.h b/gentoobrowse-api/service/notifications/xsltStreamSerializer.h new file mode 100644 index 0000000..492b954 --- /dev/null +++ b/gentoobrowse-api/service/notifications/xsltStreamSerializer.h @@ -0,0 +1,28 @@ +#ifndef ICESPIDER_CORE_XSLTSTREAMSERIALIZER_H +#define ICESPIDER_CORE_XSLTSTREAMSERIALIZER_H + +#include <slicer/xml/serializer.h> +#include <visibility.h> +#include <libxslt/transform.h> +#include <notifications.h> + +namespace Gentoo { +	class DLL_PUBLIC XsltStreamSerializer : public Slicer::XmlDocumentSerializer { +		public: +			XsltStreamSerializer(Gentoo::EmailPtr, xsltStylesheet *); +			~XsltStreamSerializer(); + +			void Serialize(Slicer::ModelPartForRootPtr mp) override; + +		protected: +			void appendHtml(xmlDoc *) const; +			void appendText(xmlDoc *) const; + +			Gentoo::EmailPtr mail; +			xmlpp::Document * doc; +			xsltStylesheet * stylesheet; +	}; +} + +#endif + diff --git a/gentoobrowse-api/unittests/Jamfile.jam b/gentoobrowse-api/unittests/Jamfile.jam index 92501c0..70fa90f 100644 --- a/gentoobrowse-api/unittests/Jamfile.jam +++ b/gentoobrowse-api/unittests/Jamfile.jam @@ -98,6 +98,14 @@ run  	: testUsers ;  run +	testNotifications.cpp +	: : : +	<dependency>../db/schema.sql +	<define>BOOST_TEST_DYN_LINK +	<library>testCommon +	: testNotifications ; + +run  	testBugs.cpp  	: : :  	<dependency>../db/schema.sql diff --git a/gentoobrowse-api/unittests/mockDefs.cpp b/gentoobrowse-api/unittests/mockDefs.cpp index 70f5b69..eeab5f2 100644 --- a/gentoobrowse-api/unittests/mockDefs.cpp +++ b/gentoobrowse-api/unittests/mockDefs.cpp @@ -2,6 +2,9 @@  #include <definedDirs.h>  Service::Service() : +	IceTray::DryIce({ +			"--GentooBrowseAPI.MailServer=smtp.random.lan:25", +		}),  	PQ::Mock("user=postgres dbname=postgres", "GentooBrowseAPI", {  			rootDir.parent_path() / "db" / "schema.sql",  			rootDir / "data.sql" }) @@ -22,6 +25,8 @@ Maintenance::Maintenance() :  TestClient::TestClient() :  	m(getProxy<Gentoo::MaintenancePrx>("maintenance")),  	p(getProxy<Gentoo::PortagePrx>("portage")), +	n(getProxy<Gentoo::NotificationsPrx>("notifications")), +	ms(getProxy<Gentoo::MailServerPrx>("mailserver")),  	u(getProxy<Gentoo::UsersPrx>("users"))  {  } diff --git a/gentoobrowse-api/unittests/mockDefs.h b/gentoobrowse-api/unittests/mockDefs.h index 9a35c1f..a0c044a 100644 --- a/gentoobrowse-api/unittests/mockDefs.h +++ b/gentoobrowse-api/unittests/mockDefs.h @@ -7,6 +7,7 @@  #include <portage.h>  #include <users.h>  #include <maintenance.h> +#include <notifications.h>  #include <selectcommandUtil.impl.h>  class DLL_PUBLIC Service : public IceTray::DryIce, PQ::Mock { @@ -25,6 +26,8 @@ class DLL_PUBLIC TestClient : public IceTray::DryIceClient {  		Gentoo::MaintenancePrx m;  		Gentoo::PortagePrx p; +		Gentoo::NotificationsPrx n; +		Gentoo::MailServerPrx ms;  		Gentoo::UsersPrx u;  }; diff --git a/gentoobrowse-api/unittests/testNotifications.cpp b/gentoobrowse-api/unittests/testNotifications.cpp new file mode 100644 index 0000000..81b28d0 --- /dev/null +++ b/gentoobrowse-api/unittests/testNotifications.cpp @@ -0,0 +1,102 @@ +#define BOOST_TEST_MODULE TestNotifications +#include <boost/test/unit_test.hpp> + +#include "mockDefs.h" +#include <fstream> + +BOOST_GLOBAL_FIXTURE( Service ); + +BOOST_FIXTURE_TEST_SUITE(tp, TestClient); + +bool +isHtml(const std::string & body) +{ +	return body.find("<html") != std::string::npos; +} + +void +save(const std::string & name, Gentoo::EmailPtr e) +{ +	std::ofstream text("/tmp/" + name + ".txt"); +	text << e->body[0]->payload; +	std::ofstream html("/tmp/" + name + ".html"); +	html << e->body[1]->payload; +} + +void +commonAssert(Gentoo::EmailPtr e) +{ +	BOOST_REQUIRE_EQUAL(e->body.size(), 2); +	BOOST_REQUIRE_EQUAL(e->body[0]->mimetype, "text/plain"); +	BOOST_REQUIRE(!isHtml(e->body[0]->payload)); +	BOOST_REQUIRE_EQUAL(e->body[1]->mimetype, "text/html"); +	BOOST_REQUIRE(isHtml(e->body[1]->payload)); +} + +#if 0 +BOOST_AUTO_TEST_CASE( testSend ) +{ +	Gentoo::NewUserPtr u = new Gentoo::NewUser(1, "testuser", "Test User", "dangoodliffe@gmail.com", "some-guid"); +	auto e = n->getSignup(u); +	BOOST_REQUIRE(e); +	ms->sendEmail(e); +} +#endif + +BOOST_AUTO_TEST_CASE( testSignup ) +{ +	Gentoo::NewUserPtr u = new Gentoo::NewUser(1, "testuser", "Test User", "test@user.com", "some-guid"); +	auto e = n->getSignup(u); +	BOOST_REQUIRE(e); +	BOOST_REQUIRE_EQUAL(e->subject, "Gentoo Browse: Welcome"); +	BOOST_REQUIRE_EQUAL(e->from.name, "Gentoo Browse"); +	BOOST_REQUIRE_EQUAL(e->from.address, "noreply@gentoobrowse.randomdan.homeip.net"); +	BOOST_REQUIRE_EQUAL(e->to.name, "Test User"); +	BOOST_REQUIRE_EQUAL(e->to.address, "test@user.com"); +	commonAssert(e); +	save("signup", e); +	for (auto p : e->body) { +		BOOST_REQUIRE(p->payload.find("Welcome to Gentoo Browse") != std::string::npos); +		BOOST_REQUIRE(p->payload.find("http://gentoobrowse.randomdan.homeip.net/user/verify/some-guid") != std::string::npos); +		BOOST_REQUIRE(p->payload.find("confirmation page: some-guid") != std::string::npos); +	} +} + +BOOST_AUTO_TEST_CASE( testNews ) +{ +	Gentoo::UserPtr u = new Gentoo::User(1, "testuser", "Test User", "test@user.com"); +	Gentoo::NewsContent nc { +		{ +			new Gentoo::Category(1, "app-test", "Test cat1"), +			new Gentoo::Category(2, "sys-test", "Test cat2"), +			new Gentoo::Category(3, "no-show", "Unused cat") +		}, +		{ +			new Gentoo::Package(1, 1, "app1", "first-seen", "Test app desc one", "Test summary 1", IceUtil::None, IceUtil::None, IceUtil::None, IceUtil::None), +			new Gentoo::Package(2, 1, "app2", "first-seen", "Test app desc two", "Test summary 1", IceUtil::None, IceUtil::None, IceUtil::None, IceUtil::None), +			new Gentoo::Package(3, 2, "app3", "first-seen", "Test app desc three", "Test summary 1", IceUtil::None, IceUtil::None, IceUtil::None, IceUtil::None), +			new Gentoo::Package(4, 2, "app4", "first-seen", "Test app no-show", "Test summary 1", IceUtil::None, IceUtil::None, IceUtil::None, IceUtil::None) +		}, +		{ +			new Gentoo::Ebuild(1, 1, 1, "1.0a", "slot", "first-seen", "last-mod", IceUtil::None), +			new Gentoo::Ebuild(2, 1, 1, "1.1a", "slot", "first-seen", "last-mod", IceUtil::None), +			new Gentoo::Ebuild(3, 1, 1, "1.2a", "slot", "first-seen", "last-mod", IceUtil::None), +			new Gentoo::Ebuild(4, 2, 1, "2.0a", "slot", "first-seen", "last-mod", IceUtil::None), +			new Gentoo::Ebuild(5, 3, 1, "3.0a", "slot", "first-seen", "last-mod", IceUtil::None), +			new Gentoo::Ebuild(6, 2, 1, "2.1a", "slot", "first-seen", "last-mod", IceUtil::None), +			new Gentoo::Ebuild(7, 3, 1, "3.1a", "slot", "first-seen", "last-mod", IceUtil::None) +		} +	}; +	auto e = n->getNews(u, nc); +	commonAssert(e); +	save("news", e); +	for (auto p : e->body) { +		BOOST_REQUIRE(p->payload.find("Latest news") != std::string::npos); +		// BOOST_REQUIRE(p->payload.find("http://gentoobrowse.randomdan.homeip.net/packages/app-test") != std::string::npos); +		// BOOST_REQUIRE(p->payload.find("http://gentoobrowse.randomdan.homeip.net/packages/app-test/app") != std::string::npos); +		BOOST_REQUIRE(p->payload.find("no-show") == std::string::npos); +	} +} + +BOOST_AUTO_TEST_SUITE_END() + | 
