cutelyst 5.0.0
A C++ Web Framework built on top of Qt, using the simple approach of Catalyst (Perl) framework.
cuteleeview.cpp
1/*
2 * SPDX-FileCopyrightText: (C) 2020-2022 Daniel Nicoletti <dantti12@gmail.com>
3 * SPDX-License-Identifier: BSD-3-Clause
4 */
5#include "action.h"
6#include "application.h"
7#include "config.h"
8#include "context.h"
9#include "cuteleeview_p.h"
10#include "cutelystcutelee.h"
11#include "response.h"
12
13#include <cutelee/metatype.h>
14#include <cutelee/qtlocalizer.h>
15
16#include <QDirIterator>
17#include <QString>
18#include <QTranslator>
19#include <QtCore/QLoggingCategory>
20
21Q_LOGGING_CATEGORY(CUTELYST_CUTELEE, "cutelyst.view.cutelee", QtWarningMsg)
22
23using namespace Cutelyst;
24
25CUTELEE_BEGIN_LOOKUP(ParamsMultiMap)
26return object.value(property);
27CUTELEE_END_LOOKUP
28
29CUTELEE_BEGIN_LOOKUP_PTR(Cutelyst::Request)
30return object->property(property.toLatin1().constData());
31CUTELEE_END_LOOKUP
32
34 : View(new CuteleeViewPrivate, parent, name)
35{
36 Q_D(CuteleeView);
37
38 Cutelee::registerMetaType<ParamsMultiMap>();
39 Cutelee::registerMetaType<Cutelyst::Request *>(); // To be able to access it's properties
40
41 d->loader = std::make_shared<Cutelee::FileSystemTemplateLoader>();
42
43 d->engine = new Cutelee::Engine(this);
44 d->engine->addTemplateLoader(d->loader);
45
46 d->initEngine();
47
48 auto app = qobject_cast<Application *>(parent);
49 if (app) {
50 // make sure templates can be found on the current directory
51 setIncludePaths({app->config(QStringLiteral("root")).toString()});
52
53 // If CUTELYST_VAR is set the template might have become
54 // {{ Cutelyst.req.base }} instead of {{ c.req.base }}
55 d->cutelystVar =
56 app->config(QStringLiteral("CUTELYST_VAR"), QStringLiteral("c")).toString();
57
58 app->loadTranslations(QStringLiteral("plugin_view_cutelee"));
59 } else {
60 // make sure templates can be found on the current directory
62 }
63}
64
65QStringList CuteleeView::includePaths() const
66{
67 Q_D(const CuteleeView);
68 return d->includePaths;
69}
70
72{
73 Q_D(CuteleeView);
74 d->loader->setTemplateDirs(paths);
75 d->includePaths = paths;
76 Q_EMIT changed();
77}
78
79QString CuteleeView::templateExtension() const
80{
81 Q_D(const CuteleeView);
82 return d->extension;
83}
84
86{
87 Q_D(CuteleeView);
88 d->extension = extension;
89 Q_EMIT changed();
90}
91
92QString CuteleeView::wrapper() const
93{
94 Q_D(const CuteleeView);
95 return d->wrapper;
96}
97
99{
100 Q_D(CuteleeView);
101 d->wrapper = name;
102 Q_EMIT changed();
103}
104
105void CuteleeView::setCache(bool enable)
106{
107 Q_D(CuteleeView);
108
109 if (enable && d->cache) {
110 return; // already enabled
111 }
112
113 delete d->engine;
114 d->engine = new Cutelee::Engine(this);
115
116 if (enable) {
117 d->cache = std::make_shared<Cutelee::CachingLoaderDecorator>(d->loader);
118 d->engine->addTemplateLoader(d->cache);
119 } else {
120 d->cache = {};
121 d->engine->addTemplateLoader(d->loader);
122 }
123 d->initEngine();
124 Q_EMIT changed();
125}
126
127Cutelee::Engine *CuteleeView::engine() const
128{
129 Q_D(const CuteleeView);
130 return d->engine;
131}
132
134{
135 Q_D(CuteleeView);
136
137 if (!isCaching()) {
138 setCache(true);
139 }
140
141 const auto includePaths = d->includePaths;
142 for (const QString &includePath : includePaths) {
143 QDirIterator it(includePath,
144 {QLatin1Char('*') + d->extension},
147 while (it.hasNext()) {
148 QString path = it.next();
149 path.remove(includePath);
150 if (path.startsWith(u'/')) {
151 path.remove(0, 1);
152 }
153
154 if (d->cache->canLoadTemplate(path)) {
155 d->cache->loadByName(path, d->engine);
156 }
157 }
158 }
159}
160
162{
163 Q_D(const CuteleeView);
164 return !!d->cache;
165}
166
168{
169 Q_D(const CuteleeView);
170
171 QByteArray ret;
172 c->setStash(d->cutelystVar, QVariant::fromValue(c));
173 const QVariantHash stash = c->stash();
174 auto it = stash.constFind(QStringLiteral("template"));
175 QString templateFile;
176 if (it != stash.constEnd()) {
177 templateFile = it.value().toString();
178 } else {
179 if (c->action() && !c->action()->reverse().isEmpty()) {
180 templateFile = c->action()->reverse() + d->extension;
181 if (templateFile.startsWith(u'/')) {
182 templateFile.remove(0, 1);
183 }
184 }
185
186 if (templateFile.isEmpty()) {
187 c->appendError(QStringLiteral(
188 "Cannot render template, template name or template stash key not defined"));
189 return ret;
190 }
191 }
192
193 qCDebug(CUTELYST_CUTELEE) << "Rendering template" << templateFile;
194
195 Cutelee::Context gc(stash);
196
197 auto localizer = std::make_shared<Cutelee::QtLocalizer>(c->locale());
198
199 auto transIt = d->translators.constFind(c->locale());
200 if (transIt != d->translators.constEnd()) {
201 localizer->installTranslator(transIt.value(), transIt.key().name());
202 }
203
204 for (const auto &[key, value] : std::as_const(d->translationCatalogs).asKeyValueRange()) {
205 localizer->loadCatalog(value, key);
206 }
207
208 gc.setLocalizer(localizer);
209
210 Cutelee::Template tmpl = d->engine->loadByName(templateFile);
211 if (tmpl->error() != Cutelee::NoError) {
212 //% "Internal server error."
213 c->res()->setBody(c->qtTrId("cutelyst-cuteleeview-err-internal-server"));
214 c->appendError(QLatin1String("Error while rendering template: ") + tmpl->errorString());
215 return ret;
216 }
217
218 QString content = tmpl->render(&gc);
219 if (tmpl->error() != Cutelee::NoError) {
220 c->res()->setBody(c->qtTrId("cutelyst-cuteleeview-err-internal-server"));
221 c->appendError(QLatin1String("Error while rendering template: ") + tmpl->errorString());
222 return ret;
223 }
224
225 if (!d->wrapper.isEmpty()) {
226 Cutelee::Template wrapper = d->engine->loadByName(d->wrapper);
227 if (tmpl->error() != Cutelee::NoError) {
228 c->res()->setBody(c->qtTrId("cutelyst-cuteleeview-err-internal-server"));
229 c->appendError(QLatin1String("Error while rendering template: ") + tmpl->errorString());
230 return ret;
231 }
232
233 Cutelee::SafeString safeContent(content, true);
234 gc.insert(QStringLiteral("content"), safeContent);
235 content = wrapper->render(&gc);
236
237 if (wrapper->error() != Cutelee::NoError) {
238 c->res()->setBody(c->qtTrId("cutelyst-cuteleeview-err-internal-server"));
239 c->appendError(QLatin1String("Error while rendering template: ") + tmpl->errorString());
240 return ret;
241 }
242 }
243
244 ret = content.toUtf8();
245 return ret;
246}
247
248void CuteleeView::addTranslator(const QLocale &locale, QTranslator *translator)
249{
250 Q_D(CuteleeView);
251 Q_ASSERT_X(translator, "add translator to CuteleeView", "invalid QTranslator object");
252 d->translators.insert(locale, translator);
253}
254
255void CuteleeView::addTranslator(const QString &locale, QTranslator *translator)
256{
257 addTranslator(QLocale(locale), translator);
258}
259
260void CuteleeView::addTranslationCatalog(const QString &path, const QString &catalog)
261{
262 Q_D(CuteleeView);
263 Q_ASSERT_X(!path.isEmpty(), "add translation catalog to CuteleeView", "empty path");
264 Q_ASSERT_X(!catalog.isEmpty(), "add translation catalog to CuteleeView", "empty catalog name");
265 d->translationCatalogs.insert(catalog, path);
266}
267
269{
270 Q_D(CuteleeView);
271 Q_ASSERT_X(!catalogs.empty(), "add translation catalogs to GranteleeView", "empty QHash");
272 d->translationCatalogs.unite(catalogs);
273}
274
276 const QString &directory,
277 const QString &prefix,
278 const QString &suffix)
279{
280 QVector<QLocale> locales;
281
282 if (Q_LIKELY(!filename.isEmpty() && !directory.isEmpty())) {
283 const QDir i18nDir(directory);
284 if (Q_LIKELY(i18nDir.exists())) {
285 const QString _prefix = prefix.isEmpty() ? QStringLiteral(".") : prefix;
286 const QString _suffix = suffix.isEmpty() ? QStringLiteral(".qm") : suffix;
287 const QStringList namesFilter =
288 QStringList({filename + _prefix + QLatin1Char('*') + _suffix});
289 const QFileInfoList tsFiles = i18nDir.entryInfoList(namesFilter, QDir::Files);
290 if (Q_LIKELY(!tsFiles.empty())) {
291 locales.reserve(tsFiles.size());
292 for (const QFileInfo &ts : tsFiles) {
293 const QString fn = ts.fileName();
294 const int prefIdx = fn.indexOf(_prefix);
295 const QString locString =
296 fn.mid(prefIdx + _prefix.length(),
297 fn.length() - prefIdx - _suffix.length() - _prefix.length());
298 QLocale loc(locString);
299 if (Q_LIKELY(loc.language() != QLocale::C)) {
300 auto trans = new QTranslator(this);
301 if (Q_LIKELY(trans->load(loc, filename, _prefix, directory))) {
302 addTranslator(loc, trans);
303 locales.append(loc);
304 qCDebug(CUTELYST_CUTELEE) << "Loaded translations for locale" << loc
305 << "from" << ts.absoluteFilePath();
306 } else {
307 delete trans;
308 qCWarning(CUTELYST_CUTELEE)
309 << "Can not load translations for locale" << loc;
310 }
311 } else {
312 qCWarning(CUTELYST_CUTELEE)
313 << "Can not load translations for invalid locale string" << locString;
314 }
315 }
316 locales.squeeze();
317 } else {
318 qCWarning(CUTELYST_CUTELEE) << "Can not find translation files for" << filename
319 << "in directory" << directory;
320 }
321 } else {
322 qCWarning(CUTELYST_CUTELEE)
323 << "Can not load translations from not existing directory:" << directory;
324 }
325 } else {
326 qCWarning(CUTELYST_CUTELEE)
327 << "Can not load translations for empty file name or empty path.";
328 }
329
330 return locales;
331}
332
333void CuteleeViewPrivate::initEngine()
334{
335 // Set also the paths from CUTELYST_PLUGINS_DIR env variable as plugin paths of cutelee engine
336 const QByteArrayList dirs = QByteArrayList{QByteArrayLiteral(CUTELYST_PLUGINS_DIR)} +
337 qgetenv("CUTELYST_PLUGINS_DIR").split(';');
338 for (const QByteArray &dir : dirs) {
339 engine->addPluginPath(QString::fromLocal8Bit(dir));
340 }
341
342 engine->insertDefaultLibrary(QStringLiteral("cutelee_cutelyst"), new CutelystCutelee(engine));
343}
344
345#include "moc_cuteleeview.cpp"
QString reverse() const noexcept
Definition component.cpp:45
QString name() const noexcept
Definition component.cpp:33
The Cutelyst Context.
Definition context.h:42
void stash(const QVariantHash &unite)
Definition context.cpp:563
QLocale locale() const noexcept
Definition context.cpp:461
Response * res() const noexcept
Definition context.cpp:104
void setStash(const QString &key, const QVariant &value)
Definition context.cpp:213
QString qtTrId(const char *id, int n=-1) const
Definition context.h:657
Action * action
Definition context.h:47
void appendError(const QString &error)
Definition context.cpp:57
A view that renders templates using Cutelee engine.
void setWrapper(const QString &name)
void addTranslator(const QLocale &locale, QTranslator *translator)
void setTemplateExtension(const QString &extension)
void addTranslationCatalogs(const QMultiHash< QString, QString > &catalogs)
QVector< QLocale > loadTranslationsFromDir(const QString &filename, const QString &directory, const QString &prefix=QStringLiteral("."), const QString &suffix=QStringLiteral(".qm"))
Cutelee::Engine * engine() const
void addTranslationCatalog(const QString &path, const QString &catalog)
QByteArray render(Context *c) const override final
void setCache(bool enable)
CuteleeView(QObject *parent=nullptr, const QString &name={})
void setIncludePaths(const QStringList &paths)
A request.
Definition request.h:42
void setBody(QIODevice *body)
Definition response.cpp:105
Abstract View component for Cutelyst.
Definition view.h:25
The Cutelyst namespace holds all public Cutelyst API.
QString currentPath()
QFileInfoList entryInfoList(QDir::Filters filters, QDir::SortFlags sort) const const
bool exists() const const
void append(QList::parameter_type value)
void reserve(qsizetype size)
void squeeze()
QLocale::Language language() const const
Q_EMITQ_EMIT
QObject * parent() const const
QString fromLocal8Bit(QByteArrayView str)
qsizetype indexOf(QChar ch, qsizetype from, Qt::CaseSensitivity cs) const const
bool isEmpty() const const
qsizetype length() const const
QString mid(qsizetype position, qsizetype n) const const
QString & remove(QChar ch, Qt::CaseSensitivity cs)
bool startsWith(QChar c, Qt::CaseSensitivity cs) const const
QByteArray toUtf8() const const
QVariant fromValue(T &&value)