[gnome-builder/wip/chergert/templates] wip on templating ideas



commit e3c004c8541ed31f25edd28bb8e93587296342f4
Author: Christian Hergert <chergert redhat com>
Date:   Wed Jan 6 04:19:43 2016 -0800

    wip on templating ideas

 configure.ac                           |    3 +
 contrib/Makefile.am                    |    1 +
 contrib/tmpl/Makefile.am               |   44 ++++
 contrib/tmpl/configure.ac.tmpl         |  203 ++++++++++++++++++
 contrib/tmpl/tmpl-branch-node.c        |  183 ++++++++++++++++
 contrib/tmpl/tmpl-branch-node.h        |   35 +++
 contrib/tmpl/tmpl-error.c              |   25 +++
 contrib/tmpl/tmpl-error.h              |   39 ++++
 contrib/tmpl/tmpl-expr-node.c          |   97 +++++++++
 contrib/tmpl/tmpl-expr-node.h          |   36 +++
 contrib/tmpl/tmpl-expr.c               |   71 ++++++
 contrib/tmpl/tmpl-expr.h               |   40 ++++
 contrib/tmpl/tmpl-lexer.c              |  183 ++++++++++++++++
 contrib/tmpl/tmpl-lexer.h              |   49 +++++
 contrib/tmpl/tmpl-node.c               |  268 +++++++++++++++++++++++
 contrib/tmpl/tmpl-node.h               |   65 ++++++
 contrib/tmpl/tmpl-parser.c             |  267 +++++++++++++++++++++++
 contrib/tmpl/tmpl-parser.h             |   48 ++++
 contrib/tmpl/tmpl-scope.c              |   57 +++++
 contrib/tmpl/tmpl-scope.h              |   35 +++
 contrib/tmpl/tmpl-template-locator.c   |  197 +++++++++++++++++
 contrib/tmpl/tmpl-template-locator.h   |   57 +++++
 contrib/tmpl/tmpl-template.c           |  368 ++++++++++++++++++++++++++++++++
 contrib/tmpl/tmpl-template.h           |   66 ++++++
 contrib/tmpl/tmpl-text-node.c          |   90 ++++++++
 contrib/tmpl/tmpl-text-node.h          |   37 ++++
 contrib/tmpl/tmpl-token-input-stream.c |  307 ++++++++++++++++++++++++++
 contrib/tmpl/tmpl-token-input-stream.h |   43 ++++
 contrib/tmpl/tmpl-token.c              |  170 +++++++++++++++
 contrib/tmpl/tmpl-token.h              |   54 +++++
 contrib/tmpl/tmpl.c                    |   74 +++++++
 contrib/tmpl/tmpl.h                    |   35 +++
 32 files changed, 3247 insertions(+), 0 deletions(-)
---
diff --git a/configure.ac b/configure.ac
index 53b5a39..8d50780 100644
--- a/configure.ac
+++ b/configure.ac
@@ -181,6 +181,8 @@ PKG_CHECK_MODULES(PYGOBJECT,[pygobject-3.0 >= pygobject_required_version],
                             [have_pygobject=no])
 PKG_CHECK_MODULES(RG,       [gtk+-3.0 >= gtk_required_version])
 PKG_CHECK_MODULES(SEARCH,   [glib-2.0 >= glib_required_version])
+PKG_CHECK_MODULES(TMPL,     [gio-2.0 >= glib_required_version
+                             gio-unix-2.0 >= glib_required_version])
 PKG_CHECK_MODULES(XML,      [gio-2.0 >= glib_required_version
                              libxml-2.0 >= libxml_required_version])
 
@@ -421,6 +423,7 @@ AC_CONFIG_FILES([
        contrib/nautilus/Makefile
        contrib/rg/Makefile
        contrib/search/Makefile
+       contrib/tmpl/Makefile
        contrib/xml/Makefile
 
        libide/ide-debug.h
diff --git a/contrib/Makefile.am b/contrib/Makefile.am
index d01f238..dcfefa4 100644
--- a/contrib/Makefile.am
+++ b/contrib/Makefile.am
@@ -6,6 +6,7 @@ SUBDIRS = \
        nautilus \
        rg \
        search \
+       tmpl \
        xml \
        $(NULL)
 
diff --git a/contrib/tmpl/Makefile.am b/contrib/tmpl/Makefile.am
new file mode 100644
index 0000000..b656fca
--- /dev/null
+++ b/contrib/tmpl/Makefile.am
@@ -0,0 +1,44 @@
+pkglibdir = $(libdir)/gnome-builder
+pkglib_LTLIBRARIES = libtemplate-glib.la
+
+libtemplate_glib_la_SOURCES = \
+       tmpl-branch-node.c \
+       tmpl-branch-node.h \
+       tmpl-error.c \
+       tmpl-error.h \
+       tmpl-expr.c \
+       tmpl-expr.h \
+       tmpl-expr-node.c \
+       tmpl-expr-node.h \
+       tmpl-lexer.c \
+       tmpl-lexer.h \
+       tmpl-node.c \
+       tmpl-node.h \
+       tmpl-parser.c \
+       tmpl-parser.h \
+       tmpl-scope.c \
+       tmpl-scope.h \
+       tmpl-template.c \
+       tmpl-template.h \
+       tmpl-template-locator.c \
+       tmpl-template-locator.h \
+       tmpl-text-node.c \
+       tmpl-text-node.h \
+       tmpl-token.c \
+       tmpl-token.h \
+       tmpl-token-input-stream.c \
+       tmpl-token-input-stream.h \
+       tmpl.h
+
+libtemplate_glib_la_LIBADD = $(TMPL_LIBS)
+libtemplate_glib_la_CFLAGS = $(TMPL_CFLAGS)
+libtemplate_glib_la_CPPFLAGS = -DTMPL_COMPILATION
+
+
+noinst_PROGRAMS = tmpl
+
+tmpl_SOURCES = tmpl.c
+tmpl_LDADD = libtemplate-glib.la
+tmpl_CFLAGS = $(TMPL_CFLAGS)
+
+-include $(top_srcdir)/git.mk
diff --git a/contrib/tmpl/configure.ac.tmpl b/contrib/tmpl/configure.ac.tmpl
new file mode 100644
index 0000000..40db4c9
--- /dev/null
+++ b/contrib/tmpl/configure.ac.tmpl
@@ -0,0 +1,203 @@
+AC_PREREQ([{{autoconf_required_version}}])
+
+dnl ***********************************************************************
+dnl Define Versioning Information
+dnl ***********************************************************************
+m4_define([major_version],[{{version_major}}])
+m4_define([minor_version],[{{version_minor}}])
+m4_define([micro_version],[{{version_micro}}])
+m4_define([version],[major_version.minor_version.micro_version])
+m4_define([interface_age],[0])
+m4_define([bugreport_url],[{{bugreport_url}}])
+m4_define([debug_default],[m4_if(m4_eval(minor_version % 2),[1],[yes],[minimum])])
+
+
+dnl ***********************************************************************
+dnl Initialize Autoconf
+dnl ***********************************************************************
+AC_INIT([{{name}}],[version],[bugreport_url])
+AC_CONFIG_HEADERS([config.h])
+AC_CONFIG_SRCDIR([configure.ac])
+AC_CONFIG_MACRO_DIR([build-aux])
+AC_CONFIG_AUX_DIR([build-aux])
+AC_SUBST(ACLOCAL_AMFLAGS, "-I build-aux")
+AC_CANONICAL_HOST
+
+
+dnl ***********************************************************************
+dnl Export version information
+dnl ***********************************************************************
+MAJOR_VERSION=major_version
+MINOR_VERSION=minor_version
+MICRO_VERSION=micro_version
+AC_SUBST(MAJOR_VERSION)
+AC_SUBST(MINOR_VERSION)
+AC_SUBST(MICRO_VERSION)
+
+
+dnl ***********************************************************************
+dnl Initialize Automake
+dnl ***********************************************************************
+AM_SILENT_RULES([yes])
+AM_INIT_AUTOMAKE([1.11 foreign subdir-objects tar-ustar no-dist-gzip dist-xz])
+AM_MAINTAINER_MODE([enable])
+
+
+dnl ***********************************************************************
+dnl Internationalization
+dnl ***********************************************************************
+IT_PROG_INTLTOOL([0.50.1])
+GETTEXT_PACKAGE=AC_PACKAGE_TARNAME
+AC_SUBST(GETTEXT_PACKAGE)
+AC_DEFINE_UNQUOTED([GETTEXT_PACKAGE], ["$GETTEXT_PACKAGE"], [GETTEXT package name])
+AM_GLIB_GNU_GETTEXT
+
+
+dnl ***********************************************************************
+dnl Check for Required Programs
+dnl ***********************************************************************
+AC_PROG_CC
+{{if enable_cplusplus}}
+AC_PROG_CXX
+{{endif}}
+AC_PROG_INSTALL
+AC_PATH_PROG([GLIB_GENMARSHAL], [glib-genmarshal])
+AC_PATH_PROG([GLIB_MKENUMS], [glib-mkenums])
+AC_PATH_PROG([GLIB_COMPILE_RESOURCES], [glib-compile-resources])
+GLIB_GSETTINGS
+PKG_PROG_PKG_CONFIG([0.22])
+{{if enable_gobject_introspection}}
+GOBJECT_INTROSPECTION_CHECK([1.42.0])
+{{endif}}
+{{if enable_vala}}
+AM_PROG_VALAC
+VAPIGEN_CHECK
+{{endif}}
+{{if enable_appstream}}
+APPSTREAM_XML
+{{endif}}
+
+
+{{if enable_c11}}
+dnl ***********************************************************************
+dnl Ensure C11 is Supported
+dnl ***********************************************************************
+AX_CHECK_COMPILE_FLAG([-std=gnu11],
+                      [CFLAGS="$CFLAGS -std=gnu11"],
+                      [AC_MSG_ERROR([C compiler cannot compile GNU C11 code])])
+
+
+{{endif}}
+dnl ***********************************************************************
+dnl Setup Debug and Tracing Support
+dnl ***********************************************************************
+AC_ARG_ENABLE(debug,
+              AS_HELP_STRING([--enable-debug=@<:@no/minimum/yes@:>@],
+                             [turn on debugging @<:@default=debug_default@:>@]),
+              ,
+              enable_debug=debug_default)
+AS_CASE(["$enable_debug"],
+        [yes],[
+            DEBUG_CFLAGS="$DEBUG_CFLAGS -O0"
+            DEBUG_CFLAGS="$DEBUG_CFLAGS -g"
+        ],
+        [minimum],[
+            DEBUG_CFLAGS="$DEBUG_CFLAGS -DG_DISABLE_CAST_CHECKS"
+        ],
+        [no],[
+            DEBUG_CFLAGS="$DEBUG_CFLAGS -DG_DISABLE_ASSERT"
+            DEBUG_CFLAGS="$DEBUG_CFLAGS -DG_DISABLE_CHECKS"
+            DEBUG_CFLAGS="$DEBUG_CFLAGS -DG_DISABLE_CAST_CHECKS"
+        ],
+        [])
+AC_SUBST(DEBUG_CFLAGS)
+
+
+dnl ***********************************************************************
+dnl Check for Required Packages
+dnl ***********************************************************************
+m4_define([glib_required_version],[{{glib_required_version}}])
+
+PKG_CHECK_MODULES({{name|strup|delimit("- ",'_')}}, [
+       gobject-2.0 >= glib_required_version
+       gio-2.0 >= glib_required_version
+])
+
+
+dnl ***********************************************************************
+dnl Initialize Libtool
+dnl ***********************************************************************
+LT_PREREQ([2.2])
+LT_INIT
+
+
+{{if enable_compiler_flags}}
+dnl ***********************************************************************
+dnl Additional C Compiler Flags
+dnl ***********************************************************************
+AX_CHECK_COMPILE_FLAG([-Werror=unknown-warning-option], [
+       ax_compiler_flags_test="-Werror=unknown-warning-option"
+], [
+       ax_compiler_flags_test=""
+])
+AX_APPEND_COMPILE_FLAGS([ \
+       -Wall \
+       -Wcast-align \
+       -Wdeclaration-after-statement \
+       -Wempty-body \
+       -Werror=format-security \
+       -Werror=format=2 \
+       -Wextra \
+       -Wformat \
+       -Wformat-nonliteral \
+       -Wformat-security \
+       -Winit-self \
+       -Wmisleading-indentation \
+       -Wmissing-include-dirs \
+       -Wshift-negative-value \
+       -Wnested-externs \
+       -Wno-missing-field-initializers \
+       -Wno-sign-compare \
+       -Wno-strict-aliasing \
+       -Wno-uninitialized \
+       -Wno-unused-parameter \
+       -Wpointer-arith \
+       -Wredundant-decls \
+       -Wreturn-type \
+       -Wshadow \
+       -Wswitch-default \
+       -Wswitch-enum \
+       -Wundef \
+       -Wuninitialized \
+], [], [$ax_compiler_flags_test])
+AC_C_CONST
+
+
+{{endif}}
+{{if enable_gtk_doc}}
+dnl ***********************************************************************
+dnl Support for gtk-doc Documentation Engine
+dnl ***********************************************************************
+GTK_DOC_CHECK([1.11],[--flavour no-tmpl])
+AM_CONDITIONAL(ENABLE_GTK_DOC, test "x$enable_gtk_doc" = "xyes")
+AC_ARG_ENABLE(doc-cross-references,
+              AS_HELP_STRING([--disable-doc-cross-references],
+                             [cross reference symbols from other libraries @<:@default=yes@:>@]),
+              enable_doc_cross_references=$enableval,
+              enable_doc_cross_references=yes)
+AM_CONDITIONAL(ENABLE_DOC_CROSS_REFERENCES, test x$enable_doc_cross_references != xno)
+
+
+{{endif}}
+dnl ***********************************************************************
+dnl Process .in Files
+dnl ***********************************************************************
+AC_CONFIG_FILES([
+       Makefile
+])
+AC_OUTPUT
+
+
+echo ""
+echo " ${PACKAGE} - ${VERSION}"
+echo ""
diff --git a/contrib/tmpl/tmpl-branch-node.c b/contrib/tmpl/tmpl-branch-node.c
new file mode 100644
index 0000000..7f46867
--- /dev/null
+++ b/contrib/tmpl/tmpl-branch-node.c
@@ -0,0 +1,183 @@
+/* tmpl-branch-node.c
+ *
+ * Copyright (C) 2016 Christian Hergert <chergert redhat com>
+ *
+ * This file is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2.1 of the License, or (at your option) any later version.
+ *
+ * This file is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+#include "tmpl-branch-node.h"
+
+struct _TmplBranchNode
+{
+  TmplNode   parent_instance;
+
+  GPtrArray  *branches;
+  GHashTable *children_by_branch;
+};
+
+/*
+ * TODO: We should probably have children nodes for if/else if/else.
+ *       The main thing is that we don't really want to accept() on
+ *       them unless we allow the lexer to unget().
+ */
+
+G_DEFINE_TYPE (TmplBranchNode, tmpl_branch_node, TMPL_TYPE_NODE)
+
+static gboolean
+tmpl_branch_node_accept (TmplNode      *node,
+                         TmplLexer     *lexer,
+                         GCancellable  *cancellable,
+                         GError       **error)
+{
+  TmplBranchNode *self = (TmplBranchNode *)node;
+
+  g_assert (TMPL_IS_BRANCH_NODE (self));
+  g_assert (lexer != NULL);
+  g_assert (!cancellable || G_IS_CANCELLABLE (cancellable));
+
+  while (TRUE)
+    {
+      TmplToken *token = NULL;
+
+      if (!tmpl_lexer_next (lexer, &token, cancellable, error))
+        return FALSE;
+
+      switch (tmpl_token_type (token))
+        {
+        case TMPL_TOKEN_ELSE:
+        case TMPL_TOKEN_ELSE_IF:
+          {
+            TmplExpr *expr;
+            const gchar *exprstr;
+
+            exprstr = tmpl_token_get_text (token);
+
+            if (!(expr = tmpl_expr_from_string (exprstr, error)))
+              return FALSE;
+
+            g_ptr_array_add (self->branches, expr);
+          }
+          break;
+
+        case TMPL_TOKEN_END:
+          return TRUE;
+
+        case TMPL_TOKEN_TEXT:
+        case TMPL_TOKEN_IF:
+        case TMPL_TOKEN_EXPRESSION:
+        case TMPL_TOKEN_FOR:
+          {
+            TmplExpr *expr;
+            TmplNode *child;
+            GPtrArray *children;
+
+            child = tmpl_node_new_for_token (token);
+            expr = g_ptr_array_index (self->branches, self->branches->len - 1);
+            children = g_hash_table_lookup (self->children_by_branch, expr);
+
+            if (children == NULL)
+              {
+                children = g_ptr_array_new_with_free_func (g_object_unref);
+                g_hash_table_insert (self->children_by_branch, expr, children);
+              }
+
+            g_ptr_array_add (children, child);
+
+            if (!tmpl_node_accept (child, lexer, cancellable, error))
+              return FALSE;
+          }
+          break;
+
+        case TMPL_TOKEN_INCLUDE:
+        default:
+          g_assert_not_reached ();
+        }
+    }
+}
+
+static void
+tmpl_branch_node_visit_children (TmplNode        *node,
+                                 TmplNodeVisitor  visitor,
+                                 gpointer         user_data)
+{
+  TmplBranchNode *self = (TmplBranchNode *)node;
+  GHashTableIter iter;
+  GPtrArray *children;
+  gint i;
+
+  g_assert (TMPL_IS_NODE (node));
+  g_assert (visitor != NULL);
+
+  g_hash_table_iter_init (&iter, self->children_by_branch);
+
+  while (g_hash_table_iter_next (&iter, NULL, (gpointer *)&children))
+    {
+      for (i = 0; i < children->len; i++)
+        {
+          TmplNode *child = g_ptr_array_index (children, i);
+
+          visitor (child, user_data);
+        }
+    }
+}
+
+static void
+tmpl_branch_node_finalize (GObject *object)
+{
+  TmplBranchNode *self = (TmplBranchNode *)object;
+
+  g_clear_pointer (&self->children_by_branch, g_hash_table_unref);
+  g_clear_pointer (&self->branches, g_ptr_array_unref);
+
+  G_OBJECT_CLASS (tmpl_branch_node_parent_class)->finalize (object);
+}
+
+static void
+tmpl_branch_node_class_init (TmplBranchNodeClass *klass)
+{
+  GObjectClass *object_class = G_OBJECT_CLASS (klass);
+  TmplNodeClass *node_class = TMPL_NODE_CLASS (klass);
+
+  object_class->finalize = tmpl_branch_node_finalize;
+
+  node_class->accept = tmpl_branch_node_accept;
+  node_class->visit_children = tmpl_branch_node_visit_children;
+}
+
+static void
+tmpl_branch_node_init (TmplBranchNode *self)
+{
+  self->branches = g_ptr_array_new_with_free_func ((GDestroyNotify)tmpl_expr_free);
+  self->children_by_branch = g_hash_table_new_full (NULL, NULL, NULL, (GDestroyNotify)g_ptr_array_unref);
+}
+
+/**
+ * tmpl_branch_node_new:
+ * @expr: (transfer full): A TmplExpr
+ *
+ * Creates a new branch (if, else if, else) using @expr for the (if)
+ * branch.
+ *
+ * Returns: (transfer full): A #TmplBranchNode.
+ */
+TmplNode *
+tmpl_branch_node_new (TmplExpr *condition)
+{
+  TmplBranchNode *self;
+
+  self = g_object_new (TMPL_TYPE_BRANCH_NODE, NULL);
+  g_ptr_array_add (self->branches, condition);
+
+  return TMPL_NODE (self);
+}
diff --git a/contrib/tmpl/tmpl-branch-node.h b/contrib/tmpl/tmpl-branch-node.h
new file mode 100644
index 0000000..e0f5efa
--- /dev/null
+++ b/contrib/tmpl/tmpl-branch-node.h
@@ -0,0 +1,35 @@
+/* tmpl-branch-node.h
+ *
+ * Copyright (C) 2016 Christian Hergert <chergert redhat com>
+ *
+ * This file is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2.1 of the License, or (at your option) any later version.
+ *
+ * This file is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+#ifndef TMPL_BRANCH_NODE_H
+#define TMPL_BRANCH_NODE_H
+
+#include "tmpl-expr.h"
+#include "tmpl-node.h"
+
+G_BEGIN_DECLS
+
+#define TMPL_TYPE_BRANCH_NODE (tmpl_branch_node_get_type())
+
+G_DECLARE_FINAL_TYPE (TmplBranchNode, tmpl_branch_node, TMPL, BRANCH_NODE, TmplNode)
+
+TmplNode *tmpl_branch_node_new (TmplExpr *condition);
+
+G_END_DECLS
+
+#endif /* TMPL_BRANCH_NODE_H */
diff --git a/contrib/tmpl/tmpl-error.c b/contrib/tmpl/tmpl-error.c
new file mode 100644
index 0000000..3306183
--- /dev/null
+++ b/contrib/tmpl/tmpl-error.c
@@ -0,0 +1,25 @@
+/* tmpl-error.c
+ *
+ * Copyright (C) 2016 Christian Hergert <chergert redhat com>
+ *
+ * This file is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2.1 of the License, or (at your option) any later version.
+ *
+ * This file is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+#include "tmpl-error.h"
+
+GQuark
+tmpl_error_quark (void)
+{
+  return g_quark_from_static_string ("tmpl-error");
+}
diff --git a/contrib/tmpl/tmpl-error.h b/contrib/tmpl/tmpl-error.h
new file mode 100644
index 0000000..fd03d8f
--- /dev/null
+++ b/contrib/tmpl/tmpl-error.h
@@ -0,0 +1,39 @@
+/* tmpl-error.h
+ *
+ * Copyright (C) 2016 Christian Hergert <chergert redhat com>
+ *
+ * This file is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2.1 of the License, or (at your option) any later version.
+ *
+ * This file is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+#ifndef TMPL_ERROR_H
+#define TMPL_ERROR_H
+
+#include <glib.h>
+
+G_BEGIN_DECLS
+
+#define TMPL_ERROR (tmpl_error_quark())
+
+typedef enum
+{
+  TMPL_ERROR_INVALID_STATE = 1,
+  TMPL_ERROR_TEMPLATE_NOT_FOUND,
+  TMPL_ERROR_CIRCULAR_INCLUDE,
+} TmplError;
+
+GQuark tmpl_error_quark (void);
+
+G_END_DECLS
+
+#endif /* TMPL_ERROR_H */
diff --git a/contrib/tmpl/tmpl-expr-node.c b/contrib/tmpl/tmpl-expr-node.c
new file mode 100644
index 0000000..f3fb91c
--- /dev/null
+++ b/contrib/tmpl/tmpl-expr-node.c
@@ -0,0 +1,97 @@
+/* tmpl-expr-node.c
+ *
+ * Copyright (C) 2016 Christian Hergert <chergert redhat com>
+ *
+ * This file is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2.1 of the License, or (at your option) any later version.
+ *
+ * This file is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+#include "tmpl-expr-node.h"
+
+struct _TmplExprNode
+{
+  TmplNode  parent_instance;
+  TmplExpr *expr;
+};
+
+G_DEFINE_TYPE (TmplExprNode, tmpl_expr_node, TMPL_TYPE_NODE)
+
+static gboolean
+tmpl_expr_node_accept (TmplNode      *node,
+                       TmplLexer     *lexer,
+                       GCancellable  *cancellable,
+                       GError       **error)
+{
+  /* no children */
+  return TRUE;
+}
+
+static void
+tmpl_expr_node_finalize (GObject *object)
+{
+  TmplExprNode *self = (TmplExprNode *)object;
+
+  g_clear_pointer (&self->expr, tmpl_expr_free);
+
+  G_OBJECT_CLASS (tmpl_expr_node_parent_class)->finalize (object);
+}
+
+static void
+tmpl_expr_node_class_init (TmplExprNodeClass *klass)
+{
+  GObjectClass *object_class = G_OBJECT_CLASS (klass);
+  TmplNodeClass *node_class = TMPL_NODE_CLASS (klass);
+
+  object_class->finalize = tmpl_expr_node_finalize;
+
+  node_class->accept = tmpl_expr_node_accept;
+}
+
+static void
+tmpl_expr_node_init (TmplExprNode *self)
+{
+}
+
+/**
+ * tmpl_expr_node_new:
+ * @expr: (transfer full): The expression
+ *
+ * Creates a new node, stealing the reference to @expr.
+ *
+ * Returns: (transfer full): A #TmplExprNode
+ */
+TmplNode *
+tmpl_expr_node_new (TmplExpr *expr)
+{
+  TmplExprNode *self;
+
+  self = g_object_new (TMPL_TYPE_EXPR_NODE, NULL);
+  self->expr = expr;
+
+  return TMPL_NODE (self);
+}
+
+/**
+ * tmpl_expr_node_get_expr:
+ *
+ * Gets the expression.
+ *
+ * Returns: (transfer none): A #TmplExpr.
+ */
+TmplExpr *
+tmpl_expr_node_get_expr (TmplExprNode *self)
+{
+  g_return_val_if_fail (TMPL_IS_EXPR_NODE (self), NULL);
+
+  return self->expr;
+}
diff --git a/contrib/tmpl/tmpl-expr-node.h b/contrib/tmpl/tmpl-expr-node.h
new file mode 100644
index 0000000..4ba2910
--- /dev/null
+++ b/contrib/tmpl/tmpl-expr-node.h
@@ -0,0 +1,36 @@
+/* tmpl-expr-node.h
+ *
+ * Copyright (C) 2016 Christian Hergert <chergert redhat com>
+ *
+ * This file is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2.1 of the License, or (at your option) any later version.
+ *
+ * This file is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+#ifndef TMPL_EXPR_NODE_H
+#define TMPL_EXPR_NODE_H
+
+#include "tmpl-expr.h"
+#include "tmpl-node.h"
+
+G_BEGIN_DECLS
+
+#define TMPL_TYPE_EXPR_NODE (tmpl_expr_node_get_type())
+
+G_DECLARE_FINAL_TYPE (TmplExprNode, tmpl_expr_node, TMPL, EXPR_NODE, TmplNode)
+
+TmplNode *tmpl_expr_node_new      (TmplExpr     *expr);
+TmplExpr *tmpl_expr_node_get_expr (TmplExprNode *self);
+
+G_END_DECLS
+
+#endif /* TMPL_EXPR_NODE_H */
diff --git a/contrib/tmpl/tmpl-expr.c b/contrib/tmpl/tmpl-expr.c
new file mode 100644
index 0000000..5472c45
--- /dev/null
+++ b/contrib/tmpl/tmpl-expr.c
@@ -0,0 +1,71 @@
+/* tmpl-expr.c
+ *
+ * Copyright (C) 2016 Christian Hergert <chergert redhat com>
+ *
+ * This file is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2.1 of the License, or (at your option) any later version.
+ *
+ * This file is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+#include "tmpl-expr.h"
+
+struct _TmplExpr
+{
+  void *dum;
+};
+
+TmplExpr *
+tmpl_expr_from_string (const gchar  *expr,
+                       GError      **error)
+{
+  TmplExpr *self;
+
+  self = g_slice_new0 (TmplExpr);
+
+  /* TODO: finish up EBNF for GJS/Python like dot expressions */
+  /* TODO: simple function calls for processing based on G-I */
+
+  return self;
+}
+
+void
+tmpl_expr_free (TmplExpr *self)
+{
+  if (self != NULL)
+    {
+      g_slice_free (TmplExpr, self);
+    }
+}
+
+gboolean
+tmpl_expr_eval (TmplExpr   *self,
+                TmplScope  *scope,
+                GValue     *return_value,
+                GError    **error)
+{
+  TmplScope *local = NULL;
+
+  g_return_val_if_fail (self != NULL, FALSE);
+  g_return_val_if_fail (return_value != NULL, FALSE);
+
+  if (scope == NULL)
+    scope = local = tmpl_scope_new ();
+
+  /* XXX: temp */
+  g_value_init (return_value, G_TYPE_BOOLEAN);
+  g_value_set_boolean (return_value, TRUE);
+
+  if (local != NULL)
+    tmpl_scope_free (scope);
+
+  return TRUE;
+}
diff --git a/contrib/tmpl/tmpl-expr.h b/contrib/tmpl/tmpl-expr.h
new file mode 100644
index 0000000..eb4b3d9
--- /dev/null
+++ b/contrib/tmpl/tmpl-expr.h
@@ -0,0 +1,40 @@
+/* tmpl-expr.h
+ *
+ * Copyright (C) 2016 Christian Hergert <chergert redhat com>
+ *
+ * This file is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2.1 of the License, or (at your option) any later version.
+ *
+ * This file is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+#ifndef TMPL_EXPR_H
+#define TMPL_EXPR_H
+
+#include <glib-object.h>
+
+#include "tmpl-scope.h"
+
+G_BEGIN_DECLS
+
+typedef struct _TmplExpr TmplExpr;
+
+TmplExpr *tmpl_expr_from_string (const gchar  *str,
+                                 GError      **error);
+gboolean  tmpl_expr_eval        (TmplExpr     *self,
+                                 TmplScope    *scope,
+                                 GValue       *return_value,
+                                 GError      **error);
+void      tmpl_expr_free        (TmplExpr     *self);
+
+G_END_DECLS
+
+#endif /* TMPL_EXPR_H */
diff --git a/contrib/tmpl/tmpl-lexer.c b/contrib/tmpl/tmpl-lexer.c
new file mode 100644
index 0000000..cdab1ab
--- /dev/null
+++ b/contrib/tmpl/tmpl-lexer.c
@@ -0,0 +1,183 @@
+/* tmpl-lexer.c
+ *
+ * Copyright (C) 2016 Christian Hergert <chergert redhat com>
+ *
+ * This file is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2.1 of the License, or (at your option) any later version.
+ *
+ * This file is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+#include "tmpl-error.h"
+#include "tmpl-lexer.h"
+#include "tmpl-template-locator.h"
+#include "tmpl-token-input-stream.h"
+
+struct _TmplLexer
+{
+  GQueue               *stream_stack;
+  TmplTemplateLocator  *locator;
+  GHashTable           *circular;
+  GSList               *unget;
+};
+
+G_DEFINE_POINTER_TYPE (TmplLexer, tmpl_lexer)
+
+TmplLexer *
+tmpl_lexer_new (GInputStream        *stream,
+                TmplTemplateLocator *locator)
+{
+  TmplLexer *self;
+
+  g_return_val_if_fail (G_IS_INPUT_STREAM (stream), NULL);
+  g_return_val_if_fail (!locator || TMPL_IS_TEMPLATE_LOCATOR (locator), NULL);
+
+  self = g_slice_new0 (TmplLexer);
+  self->stream_stack = g_queue_new ();
+  self->locator = locator ? g_object_ref (locator) : tmpl_template_locator_new ();
+  self->circular = g_hash_table_new_full (g_str_hash, g_str_equal, g_free, NULL);
+
+  g_queue_push_head (self->stream_stack, tmpl_token_input_stream_new (stream));
+
+  return self;
+}
+
+void
+tmpl_lexer_free (TmplLexer *self)
+{
+  if (self != NULL)
+    {
+      const GList *iter;
+
+      for (iter = self->stream_stack->head; iter != NULL; iter = iter->next)
+        {
+          TmplTokenInputStream *stream = iter->data;
+
+          g_object_unref (stream);
+        }
+
+      g_clear_pointer (&self->circular, g_hash_table_unref);
+      g_clear_pointer (&self->stream_stack, g_queue_free);
+      g_clear_object (&self->locator);
+      g_slice_free (TmplLexer, self);
+    }
+}
+
+/**
+ * tmpl_lexer_next:
+ * @self: A #TmplLexer.
+ * @token: (out) (transfer full): A location for a #TmplToken.
+ * @cancellable: (nullable): A #GCancellable or %NULL.
+ * @error: A location for a #GError or %NULL.
+ *
+ * Reads the next token.
+ *
+ * It is possible for %FALSE to be returned and @error left unset.
+ * This happens at the end of the stream.
+ *
+ * Returns: %TRUE if @token was set, otherwise %FALSE.
+ */
+gboolean
+tmpl_lexer_next (TmplLexer     *self,
+                 TmplToken    **token,
+                 GCancellable  *cancellable,
+                 GError       **error)
+{
+  TmplTokenInputStream *stream;
+  GError *local_error = NULL;
+  gboolean ret = FALSE;
+
+  g_return_val_if_fail (self != NULL, FALSE);
+  g_return_val_if_fail (token != NULL, FALSE);
+  g_return_val_if_fail (!cancellable || G_IS_CANCELLABLE (cancellable), FALSE);
+
+  if (self->unget != NULL)
+    {
+      *token = self->unget->data;
+      self->unget = g_slist_remove_link (self->unget, self->unget);
+      return TRUE;
+    }
+
+  while ((stream = g_queue_peek_head (self->stream_stack)))
+    {
+      if (!(*token = tmpl_token_input_stream_read_token (stream, cancellable, &local_error)))
+        {
+          /*
+           * If we finished this stream, try to move to the next one.
+           */
+          if (local_error == NULL)
+            {
+              g_queue_pop_head (self->stream_stack);
+              g_object_unref (stream);
+              continue;
+            }
+
+          goto finish;
+        }
+
+      /*
+       * If the current token is an include token, we need to resolve the
+       * include path and read tokens from it.
+       */
+      if (tmpl_token_type (*token) == TMPL_TOKEN_INCLUDE)
+        {
+          const gchar *path = tmpl_token_include_get_path (*token);
+          GInputStream *input;
+
+          if (g_hash_table_contains (self->circular, path))
+            {
+              local_error = g_error_new (TMPL_ERROR,
+                                         TMPL_ERROR_CIRCULAR_INCLUDE,
+                                         "A circular include was detected: \"%s\"",
+                                         path);
+              g_clear_pointer (token, tmpl_token_free);
+              goto finish;
+            }
+
+          if (!(input = tmpl_template_locator_locate (self->locator, path, &local_error)))
+            {
+              g_clear_pointer (token, tmpl_token_free);
+              goto finish;
+            }
+
+          g_hash_table_insert (self->circular, g_strdup (path), NULL);
+
+          stream = tmpl_token_input_stream_new (input);
+          g_queue_push_head (self->stream_stack, stream);
+
+          g_clear_pointer (token, tmpl_token_free);
+          g_object_unref (input);
+
+          continue;
+        }
+
+      ret = TRUE;
+      break;
+    }
+
+finish:
+  if ((ret == FALSE) && (local_error != NULL))
+    g_propagate_error (error, local_error);
+
+  g_assert (ret == FALSE || *token != NULL);
+
+  return ret;
+}
+
+void
+tmpl_lexer_unget (TmplLexer *self,
+                  TmplToken *token)
+{
+  g_return_if_fail (self != NULL);
+  g_return_if_fail (token != NULL);
+
+  self->unget = g_slist_prepend (self->unget, token);
+}
diff --git a/contrib/tmpl/tmpl-lexer.h b/contrib/tmpl/tmpl-lexer.h
new file mode 100644
index 0000000..8c5c9b4
--- /dev/null
+++ b/contrib/tmpl/tmpl-lexer.h
@@ -0,0 +1,49 @@
+/* tmpl-lexer.h
+ *
+ * Copyright (C) 2016 Christian Hergert <chergert redhat com>
+ *
+ * This file is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2.1 of the License, or (at your option) any later version.
+ *
+ * This file is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+#if !defined (TMPL_INSIDE) && !defined (TMPL_COMPILATION)
+# error "Only <tmpl.h> can be included directly."
+#endif
+
+#ifndef TMPL_LEXER_H
+#define TMPL_LEXER_H
+
+#include <gio/gio.h>
+
+#include "tmpl-token.h"
+#include "tmpl-template-locator.h"
+
+G_BEGIN_DECLS
+
+typedef struct _TmplLexer TmplLexer;
+
+GType      tmpl_lexer_get_type (void);
+TmplLexer *tmpl_lexer_new      (GInputStream         *stream,
+                                TmplTemplateLocator  *locator);
+void       tmpl_lexer_free     (TmplLexer            *self);
+void       tmpl_lexer_unget    (TmplLexer            *self,
+                                TmplToken            *token);
+gboolean   tmpl_lexer_next     (TmplLexer            *self,
+                                TmplToken           **token,
+                                GCancellable         *cancellable,
+                                GError              **error);
+
+
+G_END_DECLS
+
+#endif /* TMPL_LEXER_H */
diff --git a/contrib/tmpl/tmpl-node.c b/contrib/tmpl/tmpl-node.c
new file mode 100644
index 0000000..9e4799c
--- /dev/null
+++ b/contrib/tmpl/tmpl-node.c
@@ -0,0 +1,268 @@
+/* tmpl-node.c
+ *
+ * Copyright (C) 2016 Christian Hergert <chergert redhat com>
+ *
+ * This file is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2.1 of the License, or (at your option) any later version.
+ *
+ * This file is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+#include "tmpl-branch-node.h"
+#include "tmpl-expr-node.h"
+#include "tmpl-node.h"
+#include "tmpl-parser.h"
+#include "tmpl-text-node.h"
+
+typedef struct
+{
+  GPtrArray *children;
+} TmplNodePrivate;
+
+typedef struct
+{
+  GString *str;
+  gint     depth;
+} TmplNodePrintf;
+
+static void tmpl_node_printf_string (TmplNode *self,
+                                     GString  *str,
+                                     gint      depth);
+
+G_DEFINE_TYPE_WITH_PRIVATE (TmplNode, tmpl_node, G_TYPE_OBJECT)
+
+static gboolean
+tmpl_node_real_accept (TmplNode      *self,
+                       TmplLexer     *lexer,
+                       GCancellable  *cancellable,
+                       GError       **error)
+{
+  TmplNodePrivate *priv = tmpl_node_get_instance_private (self);
+  TmplToken *token = NULL;
+  TmplNode *child;
+
+  g_assert (TMPL_IS_NODE (self));
+  g_assert (lexer != NULL);
+  g_assert (!cancellable || G_IS_CANCELLABLE (cancellable));
+
+  while (TRUE)
+    {
+      if (!tmpl_lexer_next (lexer, &token, cancellable, error))
+        return FALSE;
+
+      if (priv->children == NULL)
+        priv->children = g_ptr_array_new_with_free_func (g_object_unref);
+
+      switch (tmpl_token_type (token))
+        {
+        case TMPL_TOKEN_TEXT:
+        case TMPL_TOKEN_EXPRESSION:
+        case TMPL_TOKEN_IF:
+        case TMPL_TOKEN_FOR:
+          if ((child = tmpl_node_new_for_token (token)))
+            {
+              g_ptr_array_add (priv->children, child);
+              if (!tmpl_node_accept (child, lexer, cancellable, error))
+                return FALSE;
+            }
+          break;
+
+        case TMPL_TOKEN_ELSE_IF:
+        case TMPL_TOKEN_ELSE:
+        case TMPL_TOKEN_END:
+        case TMPL_TOKEN_INCLUDE:
+        default:
+          /* XXX: Better location propagation */
+          g_warning ("Received invalid token \"%s\"",
+                     tmpl_token_get_text (token));
+          break;
+        }
+
+      tmpl_token_free (token);
+    }
+
+  return TRUE;
+}
+
+static void
+tmpl_node_real_visit_children (TmplNode        *self,
+                               TmplNodeVisitor  visitor,
+                               gpointer         user_data)
+{
+  TmplNodePrivate *priv = tmpl_node_get_instance_private (self);
+  gint i;
+
+  g_assert (TMPL_IS_NODE (self));
+  g_assert (visitor != NULL);
+
+  if (priv->children != NULL)
+    {
+      for (i = 0; i < priv->children->len; i++)
+        {
+          TmplNode *child = g_ptr_array_index (priv->children, i);
+
+          visitor (child, user_data);
+        }
+    }
+}
+
+static void
+tmpl_node_finalize (GObject *object)
+{
+  TmplNode *self = (TmplNode *)object;
+  TmplNodePrivate *priv = tmpl_node_get_instance_private (self);
+
+  g_clear_pointer (&priv->children, g_ptr_array_unref);
+
+  G_OBJECT_CLASS (tmpl_node_parent_class)->finalize (object);
+}
+
+static void
+tmpl_node_class_init (TmplNodeClass *klass)
+{
+  GObjectClass *object_class = G_OBJECT_CLASS (klass);
+
+  object_class->finalize = tmpl_node_finalize;
+
+  klass->accept = tmpl_node_real_accept;
+  klass->visit_children = tmpl_node_real_visit_children;
+}
+
+static void
+tmpl_node_init (TmplNode *self)
+{
+}
+
+TmplNode *
+tmpl_node_new (void)
+{
+  return g_object_new (TMPL_TYPE_NODE, NULL);
+}
+
+gboolean
+tmpl_node_accept (TmplNode      *self,
+                  TmplLexer     *lexer,
+                  GCancellable  *cancellable,
+                  GError       **error)
+{
+  g_return_val_if_fail (TMPL_IS_NODE (self), FALSE);
+  g_return_val_if_fail (lexer != NULL, FALSE);
+
+  return TMPL_NODE_GET_CLASS (self)->accept (self, lexer, cancellable, error);
+}
+
+void
+tmpl_node_visit_children (TmplNode        *self,
+                          TmplNodeVisitor  visitor,
+                          gpointer         user_data)
+{
+  g_return_if_fail (TMPL_IS_NODE (self));
+  g_return_if_fail (visitor != NULL);
+
+  return TMPL_NODE_GET_CLASS (self)->visit_children (self, visitor, user_data);
+}
+
+TmplNode *
+tmpl_node_new_for_token (TmplToken *token)
+{
+  g_return_val_if_fail (token != NULL, NULL);
+
+  switch (tmpl_token_type (token))
+    {
+    case TMPL_TOKEN_TEXT:
+      return tmpl_text_node_new (g_strdup (tmpl_token_get_text (token)));
+
+    case TMPL_TOKEN_IF:
+      {
+        TmplExpr *expr;
+        const gchar *exprstr;
+
+        exprstr = tmpl_token_get_text (token);
+
+        if (!(expr = tmpl_expr_from_string (exprstr, NULL)))
+          return NULL;
+
+        return tmpl_branch_node_new (expr);
+      }
+
+    case TMPL_TOKEN_FOR:
+      /* TODO */
+      return NULL;
+
+    case TMPL_TOKEN_EXPRESSION:
+      {
+        TmplExpr *expr;
+        const gchar *exprstr;
+
+        exprstr = tmpl_token_get_text (token);
+
+        if (!(expr = tmpl_expr_from_string (exprstr, NULL)))
+          return NULL;
+
+        return tmpl_expr_node_new (expr);
+      }
+
+    case TMPL_TOKEN_ELSE_IF:
+    case TMPL_TOKEN_ELSE:
+    case TMPL_TOKEN_END:
+    case TMPL_TOKEN_INCLUDE:
+    default:
+      return NULL;
+    }
+}
+
+static void
+tmpl_node_printf_visitor (TmplNode *node,
+                          gpointer  user_data)
+{
+  TmplNodePrintf *state = user_data;
+
+  g_assert (TMPL_IS_NODE (node));
+  g_assert (state != NULL);
+  g_assert (state->str != NULL);
+  g_assert (state->depth > 0);
+
+  tmpl_node_printf_string (node, state->str, state->depth);
+}
+
+static void
+tmpl_node_printf_string (TmplNode *self,
+                         GString  *str,
+                         gint      depth)
+{
+  TmplNodePrintf state = { str, depth + 1 };
+  gint i;
+
+  g_assert (TMPL_IS_NODE (self));
+  g_assert (str != NULL);
+
+  for (i = 0; i < depth; i++)
+    g_string_append (str, "  ");
+  g_string_append (str, G_OBJECT_TYPE_NAME (self));
+  g_string_append_c (str, '\n');
+
+  tmpl_node_visit_children (self,
+                            tmpl_node_printf_visitor,
+                            &state);
+}
+
+gchar *
+tmpl_node_printf (TmplNode *self)
+{
+  GString *str;
+
+  g_return_val_if_fail (TMPL_IS_NODE (self), NULL);
+
+  str = g_string_new (NULL);
+  tmpl_node_printf_string (self, str, 0);
+
+  return g_string_free (str, FALSE);
+}
diff --git a/contrib/tmpl/tmpl-node.h b/contrib/tmpl/tmpl-node.h
new file mode 100644
index 0000000..b2e59c9
--- /dev/null
+++ b/contrib/tmpl/tmpl-node.h
@@ -0,0 +1,65 @@
+/* tmpl-node.h
+ *
+ * Copyright (C) 2016 Christian Hergert <chergert redhat com>
+ *
+ * This file is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2.1 of the License, or (at your option) any later version.
+ *
+ * This file is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+#if !defined (TMPL_INSIDE) && !defined (TMPL_COMPILATION)
+# error "Only <tmpl.h> can be included directly."
+#endif
+
+#ifndef TMPL_NODE_H
+#define TMPL_NODE_H
+
+#include <gio/gio.h>
+
+#include "tmpl-lexer.h"
+
+G_BEGIN_DECLS
+
+#define TMPL_TYPE_NODE (tmpl_node_get_type())
+
+G_DECLARE_DERIVABLE_TYPE (TmplNode, tmpl_node, TMPL, NODE, GObject)
+
+typedef void (*TmplNodeVisitor) (TmplNode *self,
+                                 gpointer  user_data);
+
+struct _TmplNodeClass
+{
+  GObjectClass parent_class;
+
+  gboolean (*accept)         (TmplNode        *self,
+                              TmplLexer       *lexer,
+                              GCancellable    *cancellable,
+                              GError         **error);
+  void     (*visit_children) (TmplNode        *self,
+                              TmplNodeVisitor  visitor,
+                              gpointer         user_data);
+};
+
+TmplNode *tmpl_node_new            (void);
+TmplNode *tmpl_node_new_for_token  (TmplToken        *token);
+gboolean  tmpl_node_accept         (TmplNode         *self,
+                                    TmplLexer        *lexer,
+                                    GCancellable     *cancellable,
+                                    GError          **error);
+gchar    *tmpl_node_printf         (TmplNode         *self);
+void      tmpl_node_visit_children (TmplNode         *self,
+                                    TmplNodeVisitor   visitor,
+                                    gpointer          user_data);
+
+G_END_DECLS
+
+#endif /* TMPL_NODE_H */
diff --git a/contrib/tmpl/tmpl-parser.c b/contrib/tmpl/tmpl-parser.c
new file mode 100644
index 0000000..4942907
--- /dev/null
+++ b/contrib/tmpl/tmpl-parser.c
@@ -0,0 +1,267 @@
+/* tmpl-parser.c
+ *
+ * Copyright (C) 2016 Christian Hergert <chergert redhat com>
+ *
+ * This file is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2.1 of the License, or (at your option) any later version.
+ *
+ * This file is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+#include <glib/gi18n.h>
+
+#include "tmpl-error.h"
+#include "tmpl-lexer.h"
+#include "tmpl-node.h"
+#include "tmpl-parser.h"
+
+struct _TmplParser
+{
+  GObject               parent_instance;
+
+  TmplNode             *root;
+  GInputStream         *stream;
+  TmplTemplateLocator  *locator;
+
+  guint                 has_parsed : 1;
+};
+
+enum {
+  PROP_0,
+  PROP_LOCATOR,
+  PROP_STREAM,
+  LAST_PROP
+};
+
+G_DEFINE_TYPE (TmplParser, tmpl_parser, G_TYPE_OBJECT)
+
+static GParamSpec *properties [LAST_PROP];
+
+static void
+tmpl_parser_set_stream (TmplParser   *self,
+                        GInputStream *stream)
+{
+  g_assert (TMPL_IS_PARSER (self));
+  g_assert (!stream || G_IS_INPUT_STREAM (stream));
+
+  if (stream == NULL)
+    {
+      g_warning ("TmplParser created without a stream!");
+      return;
+    }
+
+  g_set_object (&self->stream, stream);
+}
+
+static void
+tmpl_parser_finalize (GObject *object)
+{
+  TmplParser *self = (TmplParser *)object;
+
+  g_clear_object (&self->locator);
+  g_clear_object (&self->stream);
+  g_clear_object (&self->root);
+
+  G_OBJECT_CLASS (tmpl_parser_parent_class)->finalize (object);
+}
+
+static void
+tmpl_parser_get_property (GObject    *object,
+                          guint       prop_id,
+                          GValue     *value,
+                          GParamSpec *pspec)
+{
+  TmplParser *self = TMPL_PARSER(object);
+
+  switch (prop_id)
+    {
+    case PROP_LOCATOR:
+      g_value_set_object (value, tmpl_parser_get_locator (self));
+      break;
+
+    case PROP_STREAM:
+      g_value_set_object (value, self->stream);
+      break;
+
+    default:
+      G_OBJECT_WARN_INVALID_PROPERTY_ID(object, prop_id, pspec);
+    }
+}
+
+static void
+tmpl_parser_set_property (GObject      *object,
+                          guint         prop_id,
+                          const GValue *value,
+                          GParamSpec   *pspec)
+{
+  TmplParser *self = TMPL_PARSER(object);
+
+  switch (prop_id)
+    {
+    case PROP_LOCATOR:
+      tmpl_parser_set_locator (self, g_value_get_object (value));
+      break;
+
+    case PROP_STREAM:
+      tmpl_parser_set_stream (self, g_value_get_object (value));
+      break;
+
+    default:
+      G_OBJECT_WARN_INVALID_PROPERTY_ID(object, prop_id, pspec);
+    }
+}
+
+static void
+tmpl_parser_class_init (TmplParserClass *klass)
+{
+  GObjectClass *object_class = G_OBJECT_CLASS (klass);
+
+  object_class->finalize = tmpl_parser_finalize;
+  object_class->get_property = tmpl_parser_get_property;
+  object_class->set_property = tmpl_parser_set_property;
+
+  properties [PROP_LOCATOR] =
+    g_param_spec_object ("locator",
+                         "Locator",
+                         "The template locator for resolving includes",
+                         TMPL_TYPE_TEMPLATE_LOCATOR,
+                         (G_PARAM_READWRITE | G_PARAM_EXPLICIT_NOTIFY | G_PARAM_STATIC_STRINGS));
+
+  properties [PROP_STREAM] =
+    g_param_spec_object ("stream",
+                         "Stream",
+                         "The stream to parse",
+                         G_TYPE_INPUT_STREAM,
+                         (G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY | G_PARAM_STATIC_STRINGS));
+
+  g_object_class_install_properties (object_class, LAST_PROP, properties);
+}
+
+static void
+tmpl_parser_init (TmplParser *self)
+{
+  self->root = tmpl_node_new ();
+}
+
+TmplParser *
+tmpl_parser_new (GInputStream *stream)
+{
+  g_return_val_if_fail (G_IS_INPUT_STREAM (stream), NULL);
+
+  return g_object_new (TMPL_TYPE_PARSER,
+                       "stream", stream,
+                       NULL);
+}
+
+/**
+ * tmpl_parser_get_root:
+ * @self: A #TmplNode.
+ *
+ * Gets the root node for the parser.
+ *
+ * See tmpl_parser_visit_children() to apply a visitor to all nodes created
+ * by the parser.
+ *
+ * Returns: (transfer none): An #TmplNode.
+ */
+TmplNode *
+tmpl_parser_get_root (TmplParser *self)
+{
+  g_return_val_if_fail (TMPL_IS_PARSER (self), NULL);
+
+  return self->root;
+}
+
+gboolean
+tmpl_parser_parse (TmplParser    *self,
+                   GCancellable  *cancellable,
+                   GError       **error)
+{
+  TmplLexer *lexer;
+  GError *local_error = NULL;
+
+  g_return_val_if_fail (TMPL_IS_PARSER (self), FALSE);
+  g_return_val_if_fail (!cancellable || G_IS_CANCELLABLE (cancellable), FALSE);
+
+  if (self->has_parsed)
+    {
+      g_set_error (error,
+                   TMPL_ERROR,
+                   TMPL_ERROR_INVALID_STATE,
+                   _("%s() may only be called once"),
+                   G_STRFUNC);
+      return FALSE;
+    }
+
+  self->has_parsed = TRUE;
+
+  if (self->stream == NULL)
+    {
+      g_set_error (error,
+                   TMPL_ERROR,
+                   TMPL_ERROR_INVALID_STATE,
+                   _("Parser does not contain an input stream"));
+      return FALSE;
+    }
+
+  lexer = tmpl_lexer_new (self->stream, self->locator);
+  tmpl_node_accept (self->root, lexer, cancellable, &local_error);
+  tmpl_lexer_free (lexer);
+
+  if (local_error != NULL)
+    {
+      g_propagate_error (error, local_error);
+      return FALSE;
+    }
+
+  return TRUE;
+}
+
+/**
+ * tmpl_parser_get_locator:
+ * @self: an #TmplParser
+ *
+ * Gets the template loader used for resolving includes when parsing template
+ * files.
+ *
+ * Includes are performed using the {{include "path"}} token.  The locator can
+ * be used to restrict where the templates are located from. By default, the
+ * search path is empty, and includes cannot be performed.
+ *
+ * Returns: (transfer none): A #TmplTemplateLocator.
+ */
+TmplTemplateLocator *
+tmpl_parser_get_locator (TmplParser *self)
+{
+  g_return_val_if_fail (TMPL_IS_PARSER (self), NULL);
+
+  return self->locator;
+}
+
+/**
+ * tmpl_parser_set_locator:
+ * @self: A #TmplParser
+ * @locator: A #TmplTemplateLocator
+ *
+ * Sets the template locator used to resolve {{include "path"}} directives.
+ *
+ * See tmpl_parser_get_locator() for more information.
+ */
+void
+tmpl_parser_set_locator (TmplParser          *self,
+                         TmplTemplateLocator *locator)
+{
+  g_return_if_fail (TMPL_IS_PARSER (self));
+  g_return_if_fail (TMPL_IS_TEMPLATE_LOCATOR (locator));
+
+  if (g_set_object (&self->locator, locator))
+    g_object_notify_by_pspec (G_OBJECT (self), properties [PROP_LOCATOR]);
+}
diff --git a/contrib/tmpl/tmpl-parser.h b/contrib/tmpl/tmpl-parser.h
new file mode 100644
index 0000000..da0c1dc
--- /dev/null
+++ b/contrib/tmpl/tmpl-parser.h
@@ -0,0 +1,48 @@
+/* tmpl-parser.h
+ *
+ * Copyright (C) 2016 Christian Hergert <chergert redhat com>
+ *
+ * This file is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2.1 of the License, or (at your option) any later version.
+ *
+ * This file is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+#if !defined (TMPL_INSIDE) && !defined (TMPL_COMPILATION)
+# error "Only <tmpl.h> can be included directly."
+#endif
+
+#ifndef TMPL_PARSER_H
+#define TMPL_PARSER_H
+
+#include <gio/gio.h>
+
+#include "tmpl-node.h"
+#include "tmpl-template-locator.h"
+
+G_BEGIN_DECLS
+
+#define TMPL_TYPE_PARSER (tmpl_parser_get_type())
+
+G_DECLARE_FINAL_TYPE (TmplParser, tmpl_parser, TMPL, PARSER, GObject)
+
+TmplNode            *tmpl_parser_get_root    (TmplParser           *self);
+TmplParser          *tmpl_parser_new         (GInputStream         *stream);
+TmplTemplateLocator *tmpl_parser_get_locator (TmplParser           *self);
+void                 tmpl_parser_set_locator (TmplParser           *self,
+                                              TmplTemplateLocator  *locator);
+gboolean             tmpl_parser_parse       (TmplParser           *self,
+                                              GCancellable         *cancellable,
+                                              GError              **error);
+
+G_END_DECLS
+
+#endif /* TMPL_PARSER_H */
diff --git a/contrib/tmpl/tmpl-scope.c b/contrib/tmpl/tmpl-scope.c
new file mode 100644
index 0000000..d790b6b
--- /dev/null
+++ b/contrib/tmpl/tmpl-scope.c
@@ -0,0 +1,57 @@
+/* tmpl-scope.c
+ *
+ * Copyright (C) 2016 Christian Hergert <chergert redhat com>
+ *
+ * This file is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2.1 of the License, or (at your option) any later version.
+ *
+ * This file is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+#include "tmpl-scope.h"
+
+struct _TmplScope
+{
+  GHashTable *ident_to_value;
+};
+
+TmplScope *
+tmpl_scope_new (void)
+{
+  TmplScope *self;
+
+  self = g_slice_new0 (TmplScope);
+
+  return self;
+}
+
+void
+tmpl_scope_free (TmplScope *self)
+{
+  if (self != NULL)
+    {
+      g_clear_pointer (&self->ident_to_value, g_hash_table_unref);
+      g_slice_free (TmplScope, self);
+    }
+}
+
+const GValue *
+tmpl_scope_lookup (TmplScope   *self,
+                   const gchar *identifier)
+{
+  g_return_val_if_fail (self != NULL, NULL);
+  g_return_val_if_fail (identifier != NULL, NULL);
+
+  if (self->ident_to_value != NULL)
+    return g_hash_table_lookup (self->ident_to_value, identifier);
+
+  return NULL;
+}
diff --git a/contrib/tmpl/tmpl-scope.h b/contrib/tmpl/tmpl-scope.h
new file mode 100644
index 0000000..e13f8dd
--- /dev/null
+++ b/contrib/tmpl/tmpl-scope.h
@@ -0,0 +1,35 @@
+/* tmpl-scope.h
+ *
+ * Copyright (C) 2016 Christian Hergert <chergert redhat com>
+ *
+ * This file is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2.1 of the License, or (at your option) any later version.
+ *
+ * This file is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+#ifndef TMPL_SCOPE_H
+#define TMPL_SCOPE_H
+
+#include <glib-object.h>
+
+G_BEGIN_DECLS
+
+typedef struct _TmplScope TmplScope;
+
+TmplScope    *tmpl_scope_new    (void);
+void          tmpl_scope_free   (TmplScope   *self);
+const GValue *tmpl_scope_lookup (TmplScope   *self,
+                                 const gchar *identifier);
+
+G_END_DECLS
+
+#endif /* TMPL_SCOPE_H */
diff --git a/contrib/tmpl/tmpl-template-locator.c b/contrib/tmpl/tmpl-template-locator.c
new file mode 100644
index 0000000..f31b8f5
--- /dev/null
+++ b/contrib/tmpl/tmpl-template-locator.c
@@ -0,0 +1,197 @@
+/* tmpl-template-locator.c
+ *
+ * Copyright (C) 2016 Christian Hergert <chergert redhat com>
+ *
+ * This file is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2.1 of the License, or (at your option) any later version.
+ *
+ * This file is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+#include <glib/gi18n.h>
+
+#include "tmpl-error.h"
+#include "tmpl-template-locator.h"
+
+typedef struct
+{
+  GQueue  *search_path;
+} TmplTemplateLocatorPrivate;
+
+G_DEFINE_TYPE_WITH_PRIVATE (TmplTemplateLocator, tmpl_template_locator, G_TYPE_OBJECT)
+
+static GInputStream *
+tmpl_template_locator_locate_in_path (TmplTemplateLocator *self,
+                                      const gchar         *path_base,
+                                      const gchar         *path)
+{
+  GInputStream *ret = NULL;
+  gchar *full_path;
+
+  g_assert (TMPL_IS_TEMPLATE_LOCATOR (self));
+  g_assert (path_base != NULL);
+  g_assert (path != NULL);
+
+  full_path = g_build_path (path_base, path, NULL);
+
+  if (g_str_has_prefix (full_path, "resource://"))
+    {
+      /*
+       * A mediocre attempt to prevent escapes using ../
+       */
+      if (strstr (full_path, "..") == NULL)
+        ret = g_resources_open_stream (full_path + strlen ("resource://"), 0, NULL);
+    }
+  else
+    {
+      GFile *parent = g_file_new_for_path (path_base);
+      GFile *file = g_file_new_for_path (full_path);
+      gchar *relative;
+
+      /*
+       * If the path tries to escape the search path, using ../../ or
+       * something clever, we will get an invalid path here.
+       */
+      if ((relative = g_file_get_relative_path (parent, file)))
+        {
+          g_free (relative);
+          ret = (GInputStream *)g_file_read (file, NULL, NULL);
+        }
+
+      g_object_unref (parent);
+      g_object_unref (file);
+    }
+
+  g_free (full_path);
+
+  return ret;
+}
+
+static GInputStream *
+tmpl_template_locator_real_locate (TmplTemplateLocator  *self,
+                                   const gchar          *path,
+                                   GError              **error)
+{
+  TmplTemplateLocatorPrivate *priv = tmpl_template_locator_get_instance_private (self);
+  GInputStream *ret = NULL;
+  const GList *iter;
+  const GList *search_path;
+
+  g_assert (TMPL_IS_TEMPLATE_LOCATOR (self));
+  g_assert (path != NULL);
+
+  search_path = priv->search_path->head;
+
+  for (iter = search_path; ret == NULL && iter != NULL; iter = iter->next)
+    {
+      const gchar *path_base = iter->data;
+
+      ret = tmpl_template_locator_locate_in_path (self, path_base, path);
+    }
+
+  if (ret == NULL)
+    {
+      g_set_error (error,
+                   TMPL_ERROR,
+                   TMPL_ERROR_TEMPLATE_NOT_FOUND,
+                   _("Failed to locate template \"%s\""),
+                   path);
+      return NULL;
+    }
+
+  return ret;
+}
+
+static void
+tmpl_template_locator_class_init (TmplTemplateLocatorClass *klass)
+{
+  klass->locate = tmpl_template_locator_real_locate;
+}
+
+static void
+tmpl_template_locator_init (TmplTemplateLocator *self)
+{
+  TmplTemplateLocatorPrivate *priv = tmpl_template_locator_get_instance_private (self);
+
+  priv->search_path = g_queue_new ();
+}
+
+void
+tmpl_template_locator_append_search_path (TmplTemplateLocator *self,
+                                          const gchar         *path)
+{
+  TmplTemplateLocatorPrivate *priv = tmpl_template_locator_get_instance_private (self);
+
+  g_return_if_fail (TMPL_IS_TEMPLATE_LOCATOR (self));
+  g_return_if_fail (path != NULL);
+
+  g_queue_push_tail (priv->search_path, g_strdup (path));
+}
+
+void
+tmpl_template_locator_prepend_search_path (TmplTemplateLocator *self,
+                                           const gchar         *path)
+{
+  TmplTemplateLocatorPrivate *priv = tmpl_template_locator_get_instance_private (self);
+
+  g_return_if_fail (TMPL_IS_TEMPLATE_LOCATOR (self));
+  g_return_if_fail (path != NULL);
+
+  g_queue_push_head (priv->search_path, g_strdup (path));
+}
+
+/**
+ * tmpl_template_locator_get_search_path:
+ * @self: A #TmplTemplateLocator
+ *
+ * Gets the current search path used by the template locator.
+ *
+ * Returns: (transfer full): A %NULL-terminated array of strings.
+ */
+gchar **
+tmpl_template_locator_get_search_path (TmplTemplateLocator *self)
+{
+  TmplTemplateLocatorPrivate *priv = tmpl_template_locator_get_instance_private (self);
+  GPtrArray *ar;
+  const GList *iter;
+
+  g_return_val_if_fail (TMPL_IS_TEMPLATE_LOCATOR (self), NULL);
+
+  ar = g_ptr_array_new ();
+
+  for (iter = priv->search_path->head; iter != NULL; iter = iter->next)
+    {
+      const gchar *path = iter->data;
+
+      g_ptr_array_add (ar, g_strdup (path));
+    }
+
+  g_ptr_array_add (ar, NULL);
+
+  return (gchar **)g_ptr_array_free (ar, FALSE);
+}
+
+TmplTemplateLocator *
+tmpl_template_locator_new (void)
+{
+  return g_object_new (TMPL_TYPE_TEMPLATE_LOCATOR, NULL);
+}
+
+GInputStream *
+tmpl_template_locator_locate (TmplTemplateLocator  *self,
+                              const gchar          *path,
+                              GError              **error)
+{
+  g_return_val_if_fail (TMPL_IS_TEMPLATE_LOCATOR (self), NULL);
+  g_return_val_if_fail (path != NULL, NULL);
+
+  return TMPL_TEMPLATE_LOCATOR_GET_CLASS (self)->locate (self, path, error);
+}
diff --git a/contrib/tmpl/tmpl-template-locator.h b/contrib/tmpl/tmpl-template-locator.h
new file mode 100644
index 0000000..3db60f0
--- /dev/null
+++ b/contrib/tmpl/tmpl-template-locator.h
@@ -0,0 +1,57 @@
+/* tmpl-template-locator.h
+ *
+ * Copyright (C) 2016 Christian Hergert <chergert redhat com>
+ *
+ * This file is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2.1 of the License, or (at your option) any later version.
+ *
+ * This file is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+#if !defined (TMPL_INSIDE) && !defined (TMPL_COMPILATION)
+# error "Only <tmpl.h> can be included directly."
+#endif
+
+#ifndef TMPL_TEMPLATE_LOCATOR_H
+#define TMPL_TEMPLATE_LOCATOR_H
+
+#include <gio/gio.h>
+
+#include "tmpl-template-locator.h"
+
+G_BEGIN_DECLS
+
+#define TMPL_TYPE_TEMPLATE_LOCATOR (tmpl_template_locator_get_type())
+
+G_DECLARE_DERIVABLE_TYPE (TmplTemplateLocator, tmpl_template_locator, TMPL, TEMPLATE_LOCATOR, GObject)
+
+struct _TmplTemplateLocatorClass
+{
+  GObjectClass parent_instance;
+
+  GInputStream *(*locate) (TmplTemplateLocator  *self,
+                           const gchar          *path,
+                           GError              **error);
+};
+
+TmplTemplateLocator  *tmpl_template_locator_new                 (void);
+void                  tmpl_template_locator_append_search_path  (TmplTemplateLocator  *self,
+                                                                 const gchar          *path);
+void                  tmpl_template_locator_prepend_search_path (TmplTemplateLocator  *self,
+                                                                 const gchar          *path);
+GInputStream         *tmpl_template_locator_locate              (TmplTemplateLocator  *self,
+                                                                 const gchar          *path,
+                                                                 GError              **error);
+gchar               **tmpl_template_locator_get_search_path     (TmplTemplateLocator  *self);
+
+G_END_DECLS
+
+#endif /* TMPL_TEMPLATE_LOCATOR_H */
diff --git a/contrib/tmpl/tmpl-template.c b/contrib/tmpl/tmpl-template.c
new file mode 100644
index 0000000..9f60113
--- /dev/null
+++ b/contrib/tmpl/tmpl-template.c
@@ -0,0 +1,368 @@
+/* tmpl-template.c
+ *
+ * Copyright (C) 2016 Christian Hergert <chergert redhat com>
+ *
+ * This file is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2.1 of the License, or (at your option) any later version.
+ *
+ * This file is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+#include <glib/gi18n.h>
+
+#include "tmpl-error.h"
+#include "tmpl-expr-node.h"
+#include "tmpl-parser.h"
+#include "tmpl-scope.h"
+#include "tmpl-template.h"
+#include "tmpl-text-node.h"
+
+typedef struct
+{
+  TmplParser          *parser;
+  TmplTemplateLocator *locator;
+} TmplTemplatePrivate;
+
+typedef struct
+{
+  TmplTemplate   *self;
+  TmplNode       *root;
+  GString        *output;
+  TmplScope      *scope;
+  GError        **error;
+  gboolean        result;
+} TmplTemplateExpandState;
+
+G_DEFINE_TYPE_WITH_PRIVATE (TmplTemplate, tmpl_template, G_TYPE_OBJECT)
+
+enum {
+  PROP_0,
+  PROP_LOCATOR,
+  LAST_PROP
+};
+
+static GParamSpec *properties [LAST_PROP];
+
+static void
+tmpl_template_finalize (GObject *object)
+{
+  TmplTemplate *self = (TmplTemplate *)object;
+  TmplTemplatePrivate *priv = tmpl_template_get_instance_private (self);
+
+  g_clear_object (&priv->parser);
+
+  G_OBJECT_CLASS (tmpl_template_parent_class)->finalize (object);
+}
+
+static void
+tmpl_template_get_property (GObject    *object,
+                            guint       prop_id,
+                            GValue     *value,
+                            GParamSpec *pspec)
+{
+  TmplTemplate *self = TMPL_TEMPLATE(object);
+
+  switch (prop_id)
+    {
+    case PROP_LOCATOR:
+      g_value_set_object (value, tmpl_template_get_locator (self));
+      break;
+
+    default:
+      G_OBJECT_WARN_INVALID_PROPERTY_ID(object, prop_id, pspec);
+    }
+}
+
+static void
+tmpl_template_set_property (GObject      *object,
+                            guint         prop_id,
+                            const GValue *value,
+                            GParamSpec   *pspec)
+{
+  TmplTemplate *self = TMPL_TEMPLATE(object);
+
+  switch (prop_id)
+    {
+    case PROP_LOCATOR:
+      tmpl_template_set_locator (self, g_value_get_object (value));
+      break;
+
+    default:
+      G_OBJECT_WARN_INVALID_PROPERTY_ID(object, prop_id, pspec);
+    }
+}
+
+static void
+tmpl_template_class_init (TmplTemplateClass *klass)
+{
+  GObjectClass *object_class = G_OBJECT_CLASS (klass);
+
+  object_class->finalize = tmpl_template_finalize;
+  object_class->get_property = tmpl_template_get_property;
+  object_class->set_property = tmpl_template_set_property;
+
+  properties [PROP_LOCATOR] =
+    g_param_spec_object ("locator",
+                         "Locator",
+                         "The locator used for resolving includes",
+                         TMPL_TYPE_TEMPLATE_LOCATOR,
+                         (G_PARAM_READWRITE |
+                          G_PARAM_EXPLICIT_NOTIFY |
+                          G_PARAM_CONSTRUCT |
+                          G_PARAM_STATIC_STRINGS));
+
+  g_object_class_install_properties (object_class, LAST_PROP, properties);
+}
+
+static void
+tmpl_template_init (TmplTemplate *self)
+{
+}
+
+TmplTemplate *
+tmpl_template_new (TmplTemplateLocator *locator)
+{
+  return g_object_new (TMPL_TYPE_TEMPLATE,
+                       "locator", locator,
+                       NULL);
+}
+
+gboolean
+tmpl_template_parse_file (TmplTemplate  *self,
+                          GFile         *file,
+                          GCancellable  *cancellable,
+                          GError       **error)
+{
+  GInputStream *stream;
+  gboolean ret = FALSE;
+
+  g_return_val_if_fail (TMPL_IS_TEMPLATE (self), FALSE);
+  g_return_val_if_fail (G_IS_FILE (file), FALSE);
+  g_return_val_if_fail (!cancellable || G_IS_CANCELLABLE (cancellable), FALSE);
+
+  stream = (GInputStream *)g_file_read (file, cancellable, error);
+
+  if (stream != NULL)
+    {
+      ret = tmpl_template_parse (self, stream, cancellable, error);
+      g_object_unref (stream);
+    }
+
+  return ret;
+}
+
+gboolean
+tmpl_template_parse_path (TmplTemplate  *self,
+                          const gchar   *path,
+                          GCancellable  *cancellable,
+                          GError       **error)
+{
+  gboolean ret;
+  GFile *file;
+
+  g_return_val_if_fail (TMPL_IS_TEMPLATE (self), FALSE);
+  g_return_val_if_fail (path != NULL, FALSE);
+  g_return_val_if_fail (!cancellable || G_IS_CANCELLABLE (cancellable), FALSE);
+
+  file = g_file_new_for_path (path);
+  ret = tmpl_template_parse_file (self, file, cancellable, error);
+  g_object_unref (file);
+
+  return ret;
+}
+
+gboolean
+tmpl_template_parse_resource (TmplTemplate  *self,
+                              const gchar   *resource_path,
+                              GCancellable  *cancellable,
+                              GError       **error)
+{
+  gchar *copied = NULL;
+  gboolean ret;
+  GFile *file;
+
+  g_return_val_if_fail (TMPL_IS_TEMPLATE (self), FALSE);
+  g_return_val_if_fail (resource_path != NULL, FALSE);
+  g_return_val_if_fail (!cancellable || G_IS_CANCELLABLE (cancellable), FALSE);
+
+  if (!g_str_has_prefix (resource_path, "resource://"))
+    resource_path = copied = g_strdup_printf ("resource://%s", resource_path);
+
+  file = g_file_new_for_uri (resource_path);
+  ret = tmpl_template_parse_file (self, file, cancellable, error);
+
+  g_object_unref (file);
+  g_free (copied);
+
+  return ret;
+}
+
+gboolean
+tmpl_template_parse (TmplTemplate  *self,
+                     GInputStream  *stream,
+                     GCancellable  *cancellable,
+                     GError       **error)
+{
+  TmplTemplatePrivate *priv = tmpl_template_get_instance_private (self);
+  TmplParser *parser;
+  gboolean ret = FALSE;
+
+  g_return_val_if_fail (TMPL_IS_TEMPLATE (self), FALSE);
+  g_return_val_if_fail (G_IS_INPUT_STREAM (stream), FALSE);
+  g_return_val_if_fail (!cancellable || G_IS_CANCELLABLE (cancellable), FALSE);
+
+  parser = tmpl_parser_new (stream);
+
+  tmpl_parser_set_locator (parser, priv->locator);
+
+  if (tmpl_parser_parse (parser, cancellable, error))
+    {
+      g_set_object (&priv->parser, parser);
+      ret = TRUE;
+#if 0
+      {
+        gchar *str;
+
+        str = tmpl_node_printf (tmpl_parser_get_root (parser));
+        g_print ("%s", str);
+        g_free (str);
+      }
+#endif
+    }
+
+  g_object_unref (parser);
+
+  return ret;
+}
+
+static void
+value_into_string (const GValue *value,
+                   GString      *str)
+{
+  GValue transform = G_VALUE_INIT;
+
+  g_value_init (&transform, G_TYPE_STRING);
+
+  if (g_value_transform (value, &transform))
+    {
+      const gchar *tmp;
+
+      if (NULL != (tmp = g_value_get_string (&transform)))
+        g_string_append (str, tmp);
+    }
+
+  g_value_unset (&transform);
+}
+
+static void
+tmpl_template_expand_visitor (TmplNode *node,
+                              gpointer  user_data)
+{
+  TmplTemplateExpandState *state = user_data;
+
+  g_assert (TMPL_IS_NODE (node));
+  g_assert (state != NULL);
+
+  if (TMPL_IS_TEXT_NODE (node))
+    {
+      g_string_append (state->output, tmpl_text_node_get_text (TMPL_TEXT_NODE (node)));
+    }
+  else if (TMPL_IS_EXPR_NODE (node))
+    {
+      GValue return_value = { 0 };
+      TmplExpr *expr;
+
+      expr = tmpl_expr_node_get_expr (TMPL_EXPR_NODE (node));
+
+      /* propagate error? */
+      if (!tmpl_expr_eval (expr, state->scope, &return_value, NULL))
+        return;
+
+      value_into_string (&return_value, state->output);
+      g_value_unset (&return_value);
+    }
+}
+
+gboolean
+tmpl_template_expand (TmplTemplate  *self,
+                      GOutputStream *stream,
+                      TmplScope     *scope,
+                      GCancellable  *cancellable,
+                      GError       **error)
+{
+  TmplTemplatePrivate *priv = tmpl_template_get_instance_private (self);
+  TmplTemplateExpandState state = { 0 };
+
+  g_return_val_if_fail (TMPL_IS_TEMPLATE (self), FALSE);
+  g_return_val_if_fail (G_IS_OUTPUT_STREAM (stream), FALSE);
+  g_return_val_if_fail (!cancellable || G_IS_CANCELLABLE (cancellable), FALSE);
+
+  if (priv->parser == NULL)
+    {
+      g_set_error (error,
+                   TMPL_ERROR,
+                   TMPL_ERROR_INVALID_STATE,
+                   _("Must parse template before expanding"));
+      return FALSE;
+    }
+
+  state.root = tmpl_parser_get_root (priv->parser);
+  state.self = self;
+  state.output = g_string_new (NULL);
+  state.result = TRUE;
+  state.error = error;
+  state.scope = scope;
+
+  tmpl_node_visit_children (state.root, tmpl_template_expand_visitor, &state);
+
+  if (state.result != FALSE)
+    state.result = g_output_stream_write_all (stream,
+                                              state.output->str,
+                                              state.output->len,
+                                              NULL,
+                                              cancellable,
+                                              error);
+
+  g_string_free (state.output, TRUE);
+
+  return state.result;
+}
+
+/**
+ * tmpl_template_get_locator:
+ * @template: A #TmplTemplate
+ *
+ * Gets the template locator used when resolving template includes.
+ *
+ * Returns: (transfer none): a #TmplTemplateLocator or %NULL.
+ */
+TmplTemplateLocator *
+tmpl_template_get_locator (TmplTemplate *self)
+{
+  TmplTemplatePrivate *priv = tmpl_template_get_instance_private (self);
+
+  g_return_val_if_fail (TMPL_IS_TEMPLATE (self), NULL);
+
+  return priv->locator;
+}
+
+void
+tmpl_template_set_locator (TmplTemplate        *self,
+                           TmplTemplateLocator *locator)
+{
+  TmplTemplatePrivate *priv = tmpl_template_get_instance_private (self);
+
+  g_return_if_fail (TMPL_IS_TEMPLATE (self));
+  g_return_if_fail (!locator || TMPL_IS_TEMPLATE_LOCATOR (locator));
+
+  if (g_set_object (&priv->locator, locator))
+    g_object_notify_by_pspec (G_OBJECT (self), properties [PROP_LOCATOR]);
+}
diff --git a/contrib/tmpl/tmpl-template.h b/contrib/tmpl/tmpl-template.h
new file mode 100644
index 0000000..8942f4d
--- /dev/null
+++ b/contrib/tmpl/tmpl-template.h
@@ -0,0 +1,66 @@
+/* tmpl-template.h
+ *
+ * Copyright (C) 2016 Christian Hergert <chergert redhat com>
+ *
+ * This file is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2.1 of the License, or (at your option) any later version.
+ *
+ * This file is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+#ifndef TMPL_TEMPLATE_H
+#define TMPL_TEMPLATE_H
+
+#include <gio/gio.h>
+
+#include "tmpl-scope.h"
+#include "tmpl-template-locator.h"
+
+G_BEGIN_DECLS
+
+#define TMPL_TYPE_TEMPLATE (tmpl_template_get_type())
+
+G_DECLARE_DERIVABLE_TYPE (TmplTemplate, tmpl_template, TMPL, TEMPLATE, GObject)
+
+struct _TmplTemplateClass
+{
+  GObjectClass parent_class;
+};
+
+TmplTemplate        *tmpl_template_new            (TmplTemplateLocator  *locator);
+TmplTemplateLocator *tmpl_template_get_locator    (TmplTemplate         *self);
+void                 tmpl_template_set_locator    (TmplTemplate         *self,
+                                                   TmplTemplateLocator  *locator);
+gboolean             tmpl_template_parse_file     (TmplTemplate         *self,
+                                                   GFile                *file,
+                                                   GCancellable         *cancellable,
+                                                   GError              **error);
+gboolean             tmpl_template_parse_resource (TmplTemplate         *self,
+                                                   const gchar          *path,
+                                                   GCancellable         *cancellable,
+                                                   GError              **error);
+gboolean             tmpl_template_parse_path     (TmplTemplate         *self,
+                                                   const gchar          *path,
+                                                   GCancellable         *cancellable,
+                                                   GError              **error);
+gboolean             tmpl_template_parse          (TmplTemplate         *self,
+                                                   GInputStream         *stream,
+                                                   GCancellable         *cancellable,
+                                                   GError              **error);
+gboolean             tmpl_template_expand         (TmplTemplate         *self,
+                                                   GOutputStream        *stream,
+                                                   TmplScope            *scope,
+                                                   GCancellable         *cancellable,
+                                                   GError              **error);
+
+G_END_DECLS
+
+#endif /* TMPL_TEMPLATE_H */
diff --git a/contrib/tmpl/tmpl-text-node.c b/contrib/tmpl/tmpl-text-node.c
new file mode 100644
index 0000000..8ed1972
--- /dev/null
+++ b/contrib/tmpl/tmpl-text-node.c
@@ -0,0 +1,90 @@
+/* tmpl-text-node.c
+ *
+ * Copyright (C) 2016 Christian Hergert <chergert redhat com>
+ *
+ * This file is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2.1 of the License, or (at your option) any later version.
+ *
+ * This file is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+#include "tmpl-text-node.h"
+
+struct _TmplTextNode
+{
+  TmplNode  parent_instance;
+  gchar    *text;
+};
+
+G_DEFINE_TYPE (TmplTextNode, tmpl_text_node, TMPL_TYPE_NODE)
+
+static gboolean
+tmpl_text_node_accept (TmplNode      *node,
+                       TmplLexer     *lexer,
+                       GCancellable  *cancellable,
+                       GError       **error)
+{
+  /* no children */
+  return TRUE;
+}
+
+static void
+tmpl_text_node_finalize (GObject *object)
+{
+  TmplTextNode *self = (TmplTextNode *)object;
+
+  g_clear_pointer (&self->text, g_free);
+
+  G_OBJECT_CLASS (tmpl_text_node_parent_class)->finalize (object);
+}
+
+static void
+tmpl_text_node_class_init (TmplTextNodeClass *klass)
+{
+  GObjectClass *object_class = G_OBJECT_CLASS (klass);
+  TmplNodeClass *node_class = TMPL_NODE_CLASS (klass);
+
+  object_class->finalize = tmpl_text_node_finalize;
+
+  node_class->accept = tmpl_text_node_accept;
+}
+
+static void
+tmpl_text_node_init (TmplTextNode *node)
+{
+}
+
+/**
+ * tmpl_text_node_new:
+ * @text: (transfer full): the text for the node
+ *
+ * Creates a new text node.
+ *
+ * Returns: (transfer full): the new node.
+ */
+TmplNode *
+tmpl_text_node_new (gchar *text)
+{
+  TmplTextNode *self;
+
+  self = g_object_new (TMPL_TYPE_TEXT_NODE, NULL);
+  self->text = text;
+
+  return TMPL_NODE (self);
+}
+
+const gchar *
+tmpl_text_node_get_text (TmplTextNode *self)
+{
+  g_return_val_if_fail (TMPL_IS_TEXT_NODE (self), NULL);
+
+  return self->text;
+}
diff --git a/contrib/tmpl/tmpl-text-node.h b/contrib/tmpl/tmpl-text-node.h
new file mode 100644
index 0000000..a481c9e
--- /dev/null
+++ b/contrib/tmpl/tmpl-text-node.h
@@ -0,0 +1,37 @@
+/* tmpl-text-node.h
+ *
+ * Copyright (C) 2016 Christian Hergert <chergert redhat com>
+ *
+ * This file is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2.1 of the License, or (at your option) any later version.
+ *
+ * This file is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+#ifndef TMPL_TEXT_NODE_H
+#define TMPL_TEXT_NODE_H
+
+#include "tmpl-node.h"
+
+G_BEGIN_DECLS
+
+#define TMPL_TYPE_TEXT_NODE (tmpl_text_node_get_type())
+
+G_DECLARE_FINAL_TYPE (TmplTextNode, tmpl_text_node, TMPL, TEXT_NODE, TmplNode)
+
+TmplNode     *tmpl_text_node_new      (gchar        *text);
+const gchar  *tmpl_text_node_get_text (TmplTextNode *self);
+void          tmpl_text_node_set_text (TmplTextNode *self,
+                                       const gchar  *text);
+
+G_END_DECLS
+
+#endif /* TMPL_TEXT_NODE_H */
diff --git a/contrib/tmpl/tmpl-token-input-stream.c b/contrib/tmpl/tmpl-token-input-stream.c
new file mode 100644
index 0000000..94d50bc
--- /dev/null
+++ b/contrib/tmpl/tmpl-token-input-stream.c
@@ -0,0 +1,307 @@
+/* tmpl-token-input-stream.c
+ *
+ * Copyright (C) 2016 Christian Hergert <chergert redhat com>
+ *
+ * This file is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2.1 of the License, or (at your option) any later version.
+ *
+ * This file is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+#include "tmpl-token-input-stream.h"
+
+struct _TmplTokenInputStream
+{
+  GDataInputStream parent_instance;
+};
+
+G_DEFINE_TYPE (TmplTokenInputStream, tmpl_token_input_stream, G_TYPE_DATA_INPUT_STREAM)
+
+static void
+tmpl_token_input_stream_class_init (TmplTokenInputStreamClass *klass)
+{
+}
+
+static void
+tmpl_token_input_stream_init (TmplTokenInputStream *self)
+{
+}
+
+static gboolean
+tmpl_token_input_stream_read_unichar (TmplTokenInputStream  *self,
+                                      gunichar              *unichar,
+                                      GCancellable          *cancellable,
+                                      GError               **error)
+{
+  GBufferedInputStream *stream = (GBufferedInputStream *)self;
+  gchar str[8] = { 0 };
+  gint c;
+  gint n;
+  gint i;
+
+  g_assert (TMPL_IS_TOKEN_INPUT_STREAM (self));
+  g_assert (!cancellable || G_IS_CANCELLABLE (cancellable));
+
+  if (-1 == (c = g_buffered_input_stream_read_byte (stream, cancellable, error)))
+    return FALSE;
+
+  if ((c & 0x80) == 0)
+    n = 1;
+  else if ((c & 0xE0) == 0xC0)
+    n = 2;
+  else if ((c & 0xF0) == 0xE0)
+    n = 3;
+  else if ((c & 0xF8) == 0xF0)
+    n = 4;
+  else if ((c & 0xFC) == 0xF8)
+    n = 5;
+  else if ((c & 0xFE) == 0xFC)
+    n = 6;
+  else
+    n = 0;
+
+  str [0] = c;
+
+  for (i = 1; i < n; i++)
+    {
+      if (-1 == (c = g_buffered_input_stream_read_byte (stream, cancellable, error)))
+        return FALSE;
+
+      str [i] = (gchar)c;
+    }
+
+  *unichar = g_utf8_get_char (str);
+
+  return TRUE;
+}
+
+static gchar *
+tmpl_token_input_stream_read_tag (TmplTokenInputStream  *self,
+                                  gsize                 *length,
+                                  GCancellable          *cancellable,
+                                  GError               **error)
+{
+  GBufferedInputStream *stream = (GBufferedInputStream *)self;
+  GByteArray *ar;
+  GError *local_error = NULL;
+  gboolean in_string = FALSE;
+  guchar byte;
+  gint c;
+
+  g_assert (TMPL_IS_TOKEN_INPUT_STREAM (self));
+  g_assert (length != NULL);
+  g_assert (!cancellable || G_IS_CANCELLABLE (cancellable));
+
+  ar = g_byte_array_new ();
+
+  while (TRUE)
+    {
+      if (-1 == (c = g_buffered_input_stream_read_byte (stream, cancellable, &local_error)))
+        goto failure;
+
+      switch (c)
+        {
+        case '\\':
+          if (in_string)
+            {
+              g_byte_array_append (ar, (const guchar *)"\\", 1);
+
+              if (-1 == (c = g_buffered_input_stream_read_byte (stream, cancellable, &local_error)))
+                goto failure;
+            }
+
+          break;
+
+        case '"':
+          in_string = !in_string;
+          break;
+
+        case '}':
+          if (!in_string)
+            {
+              if (-1 == (c = g_buffered_input_stream_read_byte (stream, cancellable, &local_error)))
+                goto failure;
+
+              /* Check if we got }} */
+              if (c == '}')
+                goto finish;
+
+              g_byte_array_append (ar, (const guchar *)"}", 1);
+            }
+
+          break;
+
+        default:
+          break;
+        }
+
+      byte = (guchar)c;
+      g_byte_array_append (ar, (const guchar *)&byte, 1);
+    }
+
+finish:
+  *length = ar->len;
+
+  byte = 0;
+  g_byte_array_append (ar, (const guchar *)&byte, 1);
+
+  return (gchar *)g_byte_array_free (ar, FALSE);
+
+failure:
+  *length = 0;
+
+  g_byte_array_free (ar, TRUE);
+
+  if (local_error)
+    g_propagate_error (error, local_error);
+
+  return FALSE;
+}
+
+/**
+ * tmpl_token_input_stream_read_token:
+ * @self: An #TmplTokenInputStream
+ * @error: (nullable): An optional location for a #GError
+ *
+ * Reads the next token from the underlying stream.
+ *
+ * If there was an error, %NULL is returned and @error is set.
+ *
+ * Returns: (transfer full): A #TmplToken or %NULL.
+ */
+TmplToken *
+tmpl_token_input_stream_read_token (TmplTokenInputStream  *self,
+                                    GCancellable          *cancellable,
+                                    GError               **error)
+{
+  GDataInputStream *stream = (GDataInputStream *)self;
+  GError *local_error = NULL;
+  gunichar ch;
+  gchar *text;
+  gsize len;
+
+  g_return_val_if_fail (TMPL_IS_TOKEN_INPUT_STREAM (self), NULL);
+
+  /*
+   * The syntax of the template language is very simple. All of our symbols
+   * start with {{ and end with }}. We use \ to escape, and you only ever
+   * need to escape the opening of {{ like \{{.
+   *
+   * We scan ahead until a { or \ and take appropriate action based upon
+   * peeking at the next char.
+   *
+   * Once we resolve that, we walk forward past the expression until }}.
+   * To walk past the expression, we need to know when we are in a
+   * string, since }} could theoretically be in there too.
+   */
+
+  text = g_data_input_stream_read_upto (stream, "\\{", -1, &len, cancellable, error);
+
+  /*
+   * Handle end of stream.
+   */
+  if (text == NULL)
+    return NULL;
+
+  /*
+   * Handle successful read up to \ or {.
+   */
+  if (*text != '\0')
+    return tmpl_token_new_text (text);
+
+  g_free (text);
+
+  /*
+   * Peek what type of delimiter we hit.
+   */
+  ch = g_data_input_stream_read_byte (stream, cancellable, &local_error);
+
+  if ((ch == 0) && (local_error != NULL))
+    {
+      g_propagate_error (error, local_error);
+      return NULL;
+    }
+
+  /*
+   * Handle possible escaped \{.
+   */
+  if (ch == '\\')
+    {
+      gchar str[8] = { 0 };
+
+      /*
+       * Get the next char after \.
+       */
+      if (!tmpl_token_input_stream_read_unichar (self, &ch, cancellable, error))
+        return tmpl_token_new_unichar ('\\');
+
+      /*
+       * Handle escaping {.
+       */
+      if (ch == '{')
+        return tmpl_token_new_unichar ('{');
+
+      /*
+       * Nothing escaped, return string as it was read.
+       */
+      g_unichar_to_utf8 (ch, str);
+
+      return tmpl_token_new_text (g_strdup_printf ("\\%s", str));
+    }
+
+  g_assert (ch == '{');
+
+  /*
+   * Look for { following {. If we reached the end of the stream, just
+   * return a token for the final {.
+   */
+  if (!tmpl_token_input_stream_read_unichar (self, &ch, cancellable, error))
+    return tmpl_token_new_unichar ('{');
+
+  /*
+   * If this is not a {{, then just return a string for the pair.
+   */
+  if (ch != '{')
+    {
+      gchar str[8] = { 0 };
+
+      g_unichar_to_utf8 (ch, str);
+
+      return tmpl_token_new_text (g_strdup_printf ("{%s", str));
+    }
+
+  /*
+   * Scan ahead until we find }}.
+   */
+  if (!(text = tmpl_token_input_stream_read_tag (self, &len, cancellable, error)))
+    return NULL;
+
+  return tmpl_token_new_generic (text);
+}
+
+/**
+ * tmpl_token_input_stream_new:
+ * @base_stream: the stream to read from
+ *
+ * Creates a #TmplTokenInputStream using @base_stream for the raw
+ * text stream.
+ *
+ * Returns: (transfer full): An #TmplTokenInputStream.
+ */
+TmplTokenInputStream *
+tmpl_token_input_stream_new (GInputStream *base_stream)
+{
+  g_return_val_if_fail (G_IS_INPUT_STREAM (base_stream), NULL);
+
+  return g_object_new (TMPL_TYPE_TOKEN_INPUT_STREAM,
+                       "base-stream", base_stream,
+                       NULL);
+}
diff --git a/contrib/tmpl/tmpl-token-input-stream.h b/contrib/tmpl/tmpl-token-input-stream.h
new file mode 100644
index 0000000..e951ba6
--- /dev/null
+++ b/contrib/tmpl/tmpl-token-input-stream.h
@@ -0,0 +1,43 @@
+/* tmpl-token-input-stream.h
+ *
+ * Copyright (C) 2016 Christian Hergert <chergert redhat com>
+ *
+ * This file is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2.1 of the License, or (at your option) any later version.
+ *
+ * This file is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+#if !defined (TMPL_INSIDE) && !defined (TMPL_COMPILATION)
+# error "Only <tmpl.h> can be included directly."
+#endif
+
+#ifndef TMPL_TOKEN_INPUT_STREAM_H
+#define TMPL_TOKEN_INPUT_STREAM_H
+
+#include <gio/gio.h>
+
+#include "tmpl-token.h"
+
+G_BEGIN_DECLS
+
+#define TMPL_TYPE_TOKEN_INPUT_STREAM (tmpl_token_input_stream_get_type())
+
+G_DECLARE_FINAL_TYPE (TmplTokenInputStream, tmpl_token_input_stream, TMPL, TOKEN_INPUT_STREAM, 
GDataInputStream)
+
+TmplTokenInputStream *tmpl_token_input_stream_new        (GInputStream          *base_stream);
+TmplToken            *tmpl_token_input_stream_read_token (TmplTokenInputStream  *self,
+                                                          GCancellable          *cancellable,
+                                                          GError               **error);
+
+G_END_DECLS
+
+#endif /* TMPL_TOKEN_INPUT_STREAM_H */
diff --git a/contrib/tmpl/tmpl-token.c b/contrib/tmpl/tmpl-token.c
new file mode 100644
index 0000000..6c2d4ff
--- /dev/null
+++ b/contrib/tmpl/tmpl-token.c
@@ -0,0 +1,170 @@
+/* tmpl-token.c
+ *
+ * Copyright (C) 2016 Christian Hergert <chergert redhat com>
+ *
+ * This file is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2.1 of the License, or (at your option) any later version.
+ *
+ * This file is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+#ifndef _GNU_SOURCE
+# define _GNU_SOURCE
+#endif
+
+#include <stdio.h>
+#include <stdlib.h>
+
+#include "tmpl-token.h"
+
+struct _TmplToken
+{
+  TmplTokenType type;
+  gchar *text;
+};
+
+static TmplToken *
+tmpl_token_new (void)
+{
+  TmplToken *self;
+
+  self = g_slice_new0 (TmplToken);
+
+  return self;
+}
+
+void
+tmpl_token_free (TmplToken *self)
+{
+  if (self != NULL)
+    {
+      g_free (self->text);
+      g_slice_free (TmplToken, self);
+    }
+}
+
+/**
+ * tmpl_token_new_text:
+ * @text: (transfer full): The text for the token.
+ *
+ * Creates a new #TmplToken containing @text.
+ * This is a text literal type.
+ *
+ * Returns: (transfer full): A newly allocated #TmplToken.
+ */
+TmplToken *
+tmpl_token_new_text (gchar *text)
+{
+  TmplToken *self;
+
+  self = tmpl_token_new ();
+  self->type = TMPL_TOKEN_TEXT;
+  self->text = text;
+
+  return self;
+}
+
+TmplToken *
+tmpl_token_new_unichar (gunichar ch)
+{
+  gchar utf8[8];
+  gint len;
+
+  len = g_unichar_to_utf8 (ch, utf8);
+  utf8 [len] = '\0';
+
+  return tmpl_token_new_text (g_strdup (utf8));
+}
+
+TmplTokenType
+tmpl_token_type (TmplToken *self)
+{
+  g_return_val_if_fail (self != NULL, 0);
+
+  return self->type;
+}
+
+gchar *
+tmpl_token_include_get_path (TmplToken *self)
+{
+  char *path = NULL;
+
+  g_return_val_if_fail (self != NULL, NULL);
+  g_return_val_if_fail (self->type == TMPL_TOKEN_INCLUDE, NULL);
+
+  if (1 == sscanf (self->text, "include \"%m[^\"]", &path))
+    {
+      gchar *tmp = g_strdup (path);
+      free (path);
+      return tmp;
+    }
+
+  return NULL;
+}
+
+TmplToken *
+tmpl_token_new_generic (gchar *text)
+{
+  TmplToken *self;
+
+  g_return_val_if_fail (text != NULL, NULL);
+
+  self = g_slice_new0 (TmplToken);
+
+  if (g_str_has_prefix (text, "if "))
+    {
+      self->type = TMPL_TOKEN_IF;
+      self->text = g_strstrip (g_strdup (text + 3));
+    }
+  else if (g_str_has_prefix (text, "else if "))
+    {
+      self->type = TMPL_TOKEN_ELSE_IF;
+      self->text = g_strstrip (g_strdup (text + 8));
+    }
+  else if (g_str_has_prefix (text, "else"))
+    {
+      self->type = TMPL_TOKEN_ELSE;
+      self->text = NULL;
+    }
+  else if (g_str_has_prefix (text, "end"))
+    {
+      self->type = TMPL_TOKEN_END;
+      self->text = NULL;
+    }
+  else if (g_str_has_prefix (text, "for "))
+    {
+      self->type = TMPL_TOKEN_FOR;
+      self->text = g_strstrip (g_strdup (text + 4));
+    }
+  else if (g_str_has_prefix (text, "include "))
+    {
+      self->type = TMPL_TOKEN_INCLUDE;
+      self->text = g_strstrip (g_strdup (text + 8));
+    }
+  else
+    {
+      self->type = TMPL_TOKEN_EXPRESSION;
+      self->text = g_strstrip (text);
+      text = NULL;
+    }
+
+  g_free (text);
+
+  return self;
+}
+
+const gchar *
+tmpl_token_get_text (TmplToken *self)
+{
+  g_return_val_if_fail (self != NULL, NULL);
+
+  return self->text;
+}
diff --git a/contrib/tmpl/tmpl-token.h b/contrib/tmpl/tmpl-token.h
new file mode 100644
index 0000000..6bdbfd1
--- /dev/null
+++ b/contrib/tmpl/tmpl-token.h
@@ -0,0 +1,54 @@
+/* tmpl-token.h
+ *
+ * Copyright (C) 2016 Christian Hergert <chergert redhat com>
+ *
+ * This file is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2.1 of the License, or (at your option) any later version.
+ *
+ * This file is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+#if !defined (TMPL_INSIDE) && !defined (TMPL_COMPILATION)
+# error "Only <tmpl.h> can be included directly."
+#endif
+
+#ifndef TMPL_TOKEN_H
+#define TMPL_TOKEN_H
+
+#include <glib-object.h>
+
+G_BEGIN_DECLS
+
+typedef struct _TmplToken TmplToken;
+
+typedef enum
+{
+  TMPL_TOKEN_TEXT,
+  TMPL_TOKEN_IF,
+  TMPL_TOKEN_ELSE_IF,
+  TMPL_TOKEN_ELSE,
+  TMPL_TOKEN_END,
+  TMPL_TOKEN_FOR,
+  TMPL_TOKEN_EXPRESSION,
+  TMPL_TOKEN_INCLUDE,
+} TmplTokenType;
+
+TmplToken     *tmpl_token_new_generic      (gchar     *str);
+TmplToken     *tmpl_token_new_unichar      (gunichar   ch);
+TmplToken     *tmpl_token_new_text         (gchar     *text);
+const gchar   *tmpl_token_get_text         (TmplToken *self);
+TmplTokenType  tmpl_token_type             (TmplToken *self);
+void           tmpl_token_free             (TmplToken *self);
+gchar         *tmpl_token_include_get_path (TmplToken *self);
+
+G_END_DECLS
+
+#endif /* TMPL_TOKEN_H */
diff --git a/contrib/tmpl/tmpl.c b/contrib/tmpl/tmpl.c
new file mode 100644
index 0000000..2c1d688
--- /dev/null
+++ b/contrib/tmpl/tmpl.c
@@ -0,0 +1,74 @@
+/* tmpl.c
+ *
+ * Copyright (C) 2016 Christian Hergert <chergert redhat com>
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+#include <stdlib.h>
+#include <stdio.h>
+#include <tmpl.h>
+
+#include <gio/gunixoutputstream.h>
+
+gint
+main (gint   argc,
+      gchar *argv[])
+{
+  TmplTemplateLocator *locator = NULL;
+  GOutputStream *stream = NULL;
+  TmplTemplate *tmpl = NULL;
+  TmplScope *scope = NULL;
+  GFile *file = NULL;
+  GError *error = NULL;
+  gint ret = EXIT_FAILURE;
+
+  if (argc != 2)
+    {
+      g_printerr ("usage: %s TEMPLATE\n", argv [0]);
+      return EXIT_FAILURE;
+    }
+
+  locator = tmpl_template_locator_new ();
+  tmpl_template_locator_prepend_search_path (locator, ".");
+  tmpl = tmpl_template_new (locator);
+  file = g_file_new_for_commandline_arg (argv [1]);
+  scope = tmpl_scope_new ();
+
+  if (!tmpl_template_parse_file (tmpl, file, NULL, &error))
+    {
+      g_printerr ("%s\n", error->message);
+      g_clear_error (&error);
+      goto cleanup;
+    }
+
+  stream = g_unix_output_stream_new (STDOUT_FILENO, FALSE);
+
+  if (!tmpl_template_expand (tmpl, stream, scope, NULL, &error))
+    {
+      g_printerr ("%s\n", error->message);
+      g_clear_error (&error);
+      goto cleanup;
+    }
+
+  ret = EXIT_SUCCESS;
+
+cleanup:
+  g_clear_object (&stream);
+  g_clear_object (&locator);
+  g_clear_object (&tmpl);
+  g_clear_object (&file);
+
+  return ret;
+}
diff --git a/contrib/tmpl/tmpl.h b/contrib/tmpl/tmpl.h
new file mode 100644
index 0000000..411d7de
--- /dev/null
+++ b/contrib/tmpl/tmpl.h
@@ -0,0 +1,35 @@
+/* tmpl.h
+ *
+ * Copyright (C) 2016 Christian Hergert <chergert redhat com>
+ *
+ * This file is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2.1 of the License, or (at your option) any later version.
+ *
+ * This file is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+#ifndef TMPL_H
+#define TMPL_H
+
+#include <glib.h>
+
+G_BEGIN_DECLS
+
+#define TMPL_INSIDE
+# include "tmpl-error.h"
+# include "tmpl-scope.h"
+# include "tmpl-template.h"
+# include "tmpl-template-locator.h"
+#undef TMPL_INSIDE
+
+G_END_DECLS
+
+#endif /* TMPL_H */


[Date Prev][Date Next]   [Thread Prev][Thread Next]   [Thread Index] [Date Index] [Author Index]