diff options
author | Mike Gerwitz <gerwitm@lovullo.com> | 2016-08-24 09:43:05 -0400 |
---|---|---|
committer | Mike Gerwitz <gerwitm@lovullo.com> | 2016-08-24 12:38:00 -0400 |
commit | ff01f39c1e8c9b9549d884a0db1f9a74799cf37e (patch) | |
tree | 35978db88a8d385250b1b47ad05966e19516373d /src/current/include/orderizer.xsl | |
parent | 6c0aa54bd1b7b49d736f0db3a8f48b7aa90b3b65 (diff) | |
download | tame-ff01f39c1e8c9b9549d884a0db1f9a74799cf37e.tar.gz tame-ff01f39c1e8c9b9549d884a0db1f9a74799cf37e.tar.bz2 tame-ff01f39c1e8c9b9549d884a0db1f9a74799cf37e.zip |
Liberate current implementation of "Calc DSL"
(Copyright headers will be added in the next commit; these are the
original files, unaltered in any way.)
The internal project name at LoVullo is simply "Calc DSL". This
liberates the entire thing. If anything was missed, I'll be added
later.
To continue building at LoVullo with this move, symlinks are used for
the transition; this is the exact code that is used in production.
There is a lot here---over 25,000 lines. Much of it is in disarray from
the environment surrounding its development, but it does work well for
what it was intended to do.
(LoVullo folks: fork point is 65723a0 in calcdsl.git.)
Diffstat (limited to 'src/current/include/orderizer.xsl')
-rw-r--r-- | src/current/include/orderizer.xsl | 268 |
1 files changed, 268 insertions, 0 deletions
diff --git a/src/current/include/orderizer.xsl b/src/current/include/orderizer.xsl new file mode 100644 index 0000000..170bd2b --- /dev/null +++ b/src/current/include/orderizer.xsl @@ -0,0 +1,268 @@ +<?xml version="1.0" encoding="ISO-8859-1"?> +<!-- + No, not "odorizer". + + This is a powerful system that takes an immensely complex (and insurmountable) + task out of the programmer's hands. In particular, the system: + + - Uses the map to recognize the order in which params appear in a UI + - Using that order, determines which fields in the UI could potentially + cause fields that appear previous to them to become invalid (e.g. hide) + due to classification criteria. + - Based on those conflicts, constructs a custom classification that ignores + only the conflicting portions, allowing a great deal of precision in + controlling field validity up to that particular point in the UI. + + The result is highly-refined, custom-generated classifications per-question + for only the criteria that are needed to ensure that fields cannot hide fields + previous to them: A task that would otherwise be prohibitively complicated (or + would otherwise have to be far too coarse) if done manually in systems with + highly sophisticated classification schemes. Furthermore, the result is wholly + deterministic. + + It should be noted that futher benefit can be realized when the map is altered + or questions in the UI are reordered; the system will simply re-calculate new + classifications that yield desirable results. +--> + +<xsl:stylesheet version="2.0" + xmlns="http://www.w3.org/1999/xhtml" + xmlns:xsl="http://www.w3.org/1999/XSL/Transform" + xmlns:lv="http://www.lovullo.com/rater" + xmlns:lvp="http://www.lovullo.com" + xmlns:lvm="http://www.lovullo.com/rater/map" + xmlns:lvmc="http://www.lovullo.com/rater/map/compiler" + xmlns:gc="http://www.lovullo.com/calc/global-classifier" + xmlns:c="http://www.lovullo.com/calc" + xmlns:o="http://www.lovullo.com/rater/compiler/orderizer" + xmlns:preproc="http://www.lovullo.com/rater/preproc"> + + + +<!-- + Determines param conflicts based on UI ordering +--> +<xsl:template name="o:analyze"> + <xsl:param name="class-deps" /> + <xsl:param name="ui" /> + <xsl:param name="sdmap" /> + <xsl:param name="param-classes" /> + + <xsl:variable name="rater" select="." /> + + <!-- put fields in the order that they appear in the UI --> + <!-- TODO: duplicate check (error) --> + <xsl:variable name="ordered"> + <o:ordered> + <xsl:for-each select="$ui//lvp:question"> + <xsl:variable name="id" select="@id" /> + <xsl:copy-of select="$sdmap/lvmc:map[ @from=$id ]" /> + </xsl:for-each> + </o:ordered> + </xsl:variable> + + <!-- the param-class list gives us the classes that are directly used; let's + get a list of all classes that are used by those classes as well, which + will simplify the queries to come --> + <xsl:message>[orderizer] recursively discovering param classes...</xsl:message> + <xsl:variable name="param-classes-expanded"> + <xsl:apply-templates select="$param-classes" mode="o:gen-param-reflist"> + <xsl:with-param name="class-deps" select="$class-deps" /> + </xsl:apply-templates> + </xsl:variable> + + <xsl:variable name="initial"> + <o:analysis> + <xsl:for-each select="$ordered//lvmc:*"> + <xsl:variable name="cur" select="." /> + <xsl:variable name="pref" select="@to" /> + + <!-- progress indicator --> + <xsl:message> + <xsl:text>[orderizer] checking previous UI siblings: </xsl:text> + <xsl:value-of select="@to" /> + </xsl:message> + + <!-- preceding --> + <xsl:for-each select=" + $param-classes/gc:param[ + @ref=$cur/preceding-sibling::lvmc:map/@to + ]"> + + <!-- immediate conflicts (we check these separately rather than in the + xpath above, which would be cleaner, so that they can be + processed --> + <xsl:variable name="conflicts" select=" + .//gc:class[ + @ref=$param-classes-expanded/o:param-refs/o:param[ + @ref=$pref + ]/o:class/@ref + ] + " /> + + <xsl:if test="$conflicts"> + <o:conflict ref="{@ref}" relative-to="{$pref}"> + <xsl:for-each select="$conflicts"> + <!-- record the immediate problem --> + <o:class ref="{@ref}" yields="{@yields}" /> + </xsl:for-each> + </o:conflict> + </xsl:if> + </xsl:for-each> + </xsl:for-each> + </o:analysis> + </xsl:variable> + + <xsl:apply-templates select="$initial/o:analysis" mode="o:reduce-analysis" /> +</xsl:template> + + +<xsl:template match="gc:param-classes" mode="o:gen-param-reflist" priority="5"> + <xsl:param name="class-deps" /> + + <o:param-refs> + <xsl:apply-templates select="gc:param" mode="o:gen-param-reflist"> + <xsl:with-param name="class-deps" select="$class-deps" /> + </xsl:apply-templates> + </o:param-refs> +</xsl:template> + +<xsl:template match="gc:param" mode="o:gen-param-reflist" priority="5"> + <xsl:param name="class-deps" /> + + <o:param ref="{@ref}"> + <xsl:apply-templates select="gc:class" mode="o:gen-param-reflist"> + <xsl:with-param name="class-deps" select="$class-deps" /> + </xsl:apply-templates> + </o:param> +</xsl:template> + +<xsl:template match="gc:class" mode="o:gen-param-reflist" priority="5"> + <xsl:param name="class-deps" /> + + <!-- well, this class is certainly used --> + <o:class ref="{@ref}" /> + + <xsl:variable name="ref" select="@ref" /> + + <!-- but how about things that use this class? --> + <xsl:apply-templates select="$class-deps/preproc:class[ @ref=$ref ]" + mode="o:gen-param-reflist"> + <xsl:with-param name="class-deps" select="$class-deps" /> + </xsl:apply-templates> +</xsl:template> + +<xsl:template match="preproc:class" mode="o:gen-param-reflist" priority="5"> + <xsl:param name="class-deps" /> + + <o:class ref="{@ref}" /> + + <xsl:apply-templates mode="o:gen-param-reflist"> + <xsl:with-param name="class-deps" select="$class-deps" /> + </xsl:apply-templates> +</xsl:template> + +<xsl:template match="*" mode="o:gen-param-reflist" priority="1"> + <!-- do nothing --> +</xsl:template> + + +<xsl:template match="preproc:class" mode="o:get-indirect-params" priority="5"> + <xsl:apply-templates mode="o:get-indirect-params" /> +</xsl:template> + +<xsl:template match="preproc:class-dep" mode="o:get-indirect-params" priority="5"> + <xsl:variable name="ref" select="@ref" /> + + <xsl:apply-templates + select="ancestor::preproc:class-deps/preproc:class[ @ref=$ref ]" + mode="o:get-indirect-params" /> +</xsl:template> + +<xsl:template match="preproc:rate-dep" mode="o:get-indirect-params" priority="5"> + <xsl:variable name="ref" select="@ref" /> + + <xsl:apply-templates + select="//preproc:rate-deps/preproc:rate[ @ref=$ref ]" + mode="o:get-indirect-params" /> +</xsl:template> + +<xsl:template match="preproc:param" mode="o:get-indirect-params" priority="5"> + <xsl:copy-of select="." /> +</xsl:template> + +<xsl:template match="*" mode="o:get-indirect-params" priority="1"> + <!-- do nothing --> +</xsl:template> + + +<!-- + Combines initial analysis results, merging common refs and de-duplicating + conflicts. +--> +<xsl:template match="o:analysis" mode="o:reduce-analysis"> + <!-- start by combining the contents of all unique refs --> + <xsl:variable name="combined"> + <xsl:for-each select=" + o:conflict[ + not( @ref=preceding-sibling::o:conflict/@ref ) + ] + "> + + <xsl:variable name="ref" select="@ref" /> + <xsl:variable name="dups" select=" + following-sibling::o:conflict[ @ref=$ref ] + " /> + + <o:conflict ref="{@ref}"> + <!-- add relativity nodes --> + <xsl:for-each select="$dups|."> + <o:relative-to ref="{@relative-to}" /> + </xsl:for-each> + + <!-- add class deps --> + <o:reasons> + <xsl:copy-of select="($dups|.)/o:class" /> + </o:reasons> + </o:conflict> + </xsl:for-each> + </xsl:variable> + + <!-- now, remove duplicates resulting from the above operation --> + <xsl:copy> + <xsl:copy-of select="@*" /> + + <xsl:apply-templates select="$combined/o:conflict" + mode="o:reduce-analysis" /> + </xsl:copy> +</xsl:template> + +<xsl:template match="o:conflict" mode="o:reduce-analysis"> + <xsl:copy> + <xsl:copy-of select="@*" /> + + <!-- unique relative-to nodes --> + <xsl:copy-of select=" + o:relative-to[ + not( @ref=preceding-sibling::o:relative-to/@ref ) + ] + " /> + + <xsl:apply-templates select="o:reasons" mode="o:reduce-analysis" /> + </xsl:copy> +</xsl:template> + +<xsl:template match="o:reasons" mode="o:reduce-analysis"> + <xsl:copy> + <xsl:copy-of select="@*" /> + + <!-- unique reasons --> + <xsl:copy-of select=" + o:class[ + not( @ref=preceding-sibling::o:class/@ref ) + ] + " /> + </xsl:copy> +</xsl:template> + +</xsl:stylesheet> |