From: Lady <redacted>
Date: Thu, 8 Feb 2024 03:41:33 +0000 (-0500)
Subject: Refactor transforms & add 书社:application stage
X-Git-Tag: 0.6.0
X-Git-Url: https://git.ladys.computer/Shushe/commitdiff_plain/d925cf1fa09c85a10f150a643579b44d08b4626a?hp=d64d93fe12371f87cc42d72e7e874115b28fb252

Refactor transforms & add 书社:application stage

The main goal of this commit was to add a
`<书社:apply-attributes-to-root>` element, to allow transforms to pass
attributes up to the root element of the result, for example `@lang`
information. This required an extensive refactor of a lot of the
transform infrastructure and the creation of a new transform stage,
`书社:application`, which follows the ordinary transform and solely
handles `<书社:apply-attributes-to-root>` and
`<书社:apply-attributes>`. Other `@书社:*` attributes are removed at
this stage, but it isn’t generally recommended that transforms try to
hook in here.

This commit also makes a number of smaller changes :⁠—

- Use `node()` in place of element wildcards anywhere where
  specifically only matching elements wasn’t intended.

  In particular, even in places where text is not expected, there may
  be comments to preserve.

- Only add `@itemscope` and `@itemtype` attributes to H·T·M·L elements,
  since they are only defined for elements in that namespace.

- Provide `@书社:identifier` on documents and embeds to get the
  `about:shushe` u·r·i of the resource.

- In transforms which generate transforms, `<xsla:text>` elements which
  provide only white·space need `<xslt:text>` children to ensure the
  whitespace isn’t stripped. (Note: In the actual source text, `xsla:`
  is given the `xslt:` prefix and `xslt:` is the default prefix.)
  Similarly, it’s necessary to provide attribute value templates using
  a `<xslt:attribute>` element rather than with the literal result
  element syntax, to prevent them from being prematurely applied.
---

diff --git a/GNUmakefile b/GNUmakefile
index f0c9e8a..1697903 100644
--- a/GNUmakefile
+++ b/GNUmakefile
@@ -497,7 +497,7 @@ $(BUILDDIR)/parser.xslt : $(BUILDDIR)/parser.catalog $(THISDIR)/lib/catalog2pars
 $(call parsed,$(sourcefiles) $(sourceincludes)) : % : $$(call unparsed,$$@) $(BUILDDIR)/parser.xslt $(typeupdates)
 	@$(PRINTF) '%s\n' $(call quote,Processing `$<´…)
 	$(silent)$(call ensuredirectory,$(dir $@))
-	$(silent)$(if $(filter $<,$(assetfiles)),$(PRINTF) '%s\n' $(call quote,<object xmlns="http://www.w3.org/1999/xhtml" type="$(call typeoffile,$<)" data="$(call datauri,$<)"/>) > $(call quote,$@),$(if $(filter $<,$(plaintextfiles)),$(call wrapplaintext,$<),$(CAT) $(call quote,$<)) | $(XSLTPROC) -o $(call quote,$@) --stringparam BUILDTIME $$(TZ= $(DATE) '+%Y-%m-%dT%H:%M:%SZ') --stringparam SRCTIME $$(TZ= $(STAT) -f '%Sm' -t '%Y-%m-%dT%H:%M:%SZ' $(call quote,$<)) --stringparam CKSUM $$($(CKSUM) $(call quote,$<) | $(SED) 's/[ ].*//')$(if $(THISREV), --stringparam THISREV $(call quote,$(THISREV)),)$(if $(SRCREV), --stringparam SRCREV $(call quote,$(SRCREV)),) $(call quote,$(BUILDDIR)/parser.xslt) -)
+	$(silent)$(if $(filter $<,$(assetfiles)),$(PRINTF) '%s\n' $(call quote,<object xmlns="http://www.w3.org/1999/xhtml" type="$(call typeoffile,$<)" data="$(call datauri,$<)"/>) > $(call quote,$@),$(if $(filter $<,$(plaintextfiles)),$(call wrapplaintext,$<),$(CAT) $(call quote,$<)) | $(XSLTPROC) -o $(call quote,$@) --stringparam BUILDTIME $$(TZ= $(DATE) '+%Y-%m-%dT%H:%M:%SZ') --stringparam IDENTIFIER $(call quote,$(call localuri,$<)) --stringparam SRCTIME $$(TZ= $(STAT) -f '%Sm' -t '%Y-%m-%dT%H:%M:%SZ' $(call quote,$<)) --stringparam CKSUM $$($(CKSUM) $(call quote,$<) | $(SED) 's/[ ].*//')$(if $(THISREV), --stringparam THISREV $(call quote,$(THISREV)),)$(if $(SRCREV), --stringparam SRCREV $(call quote,$(SRCREV)),) $(call quote,$(BUILDDIR)/parser.xslt) -)
 
 # Generate a catalog of all parsed files, for use when processing includes.
 #
@@ -533,7 +533,7 @@ $(BUILDDIR)/transform.xslt : $(BUILDDIR)/transform.catalog $(THISDIR)/lib/catalo
 $(call compiled,$(compilablefiles)) : $(BUILDDIR)/public/% : $$(call parsed,$$(call uncompiled,$$@)) $(BUILDDIR)/transform.xslt $$(call parsed,$$(call dependencies,$$(call uncompiled,$$@)))
 	$(silent)$(call ensuredirectory,$(dir $@))
 	@$(PRINTF) '%s\n' $(call quote,Compiling </$*>…)
-	$(silent)$(XSLTPROC) -o $(call quote,$@) --stringparam CATALOG 'catalog' --stringparam BUILDTIME $$(TZ= $(DATE) '+%Y-%m-%dT%H:%M:%SZ') --stringparam SRCTIME $$(TZ= $(STAT) -f '%Sm' -t '%Y-%m-%dT%H:%M:%SZ' $(call quote,$(call uncompiled,$@))) --stringparam PATH $(call quote,/$*) --stringparam CKSUM $$($(CKSUM) $(call quote,$(call uncompiled,$@)) | $(SED) 's/[ ].*//')$(if $(THISREV), --stringparam THISREV $(call quote,$(THISREV)),)$(if $(SRCREV), --stringparam SRCREV $(call quote,$(SRCREV)),) $(call quote,$(BUILDDIR)/transform.xslt) $(call quote,$<)
+	$(silent)$(XSLTPROC) -o $(call quote,$@) --stringparam CATALOG 'catalog' --stringparam BUILDTIME $$(TZ= $(DATE) '+%Y-%m-%dT%H:%M:%SZ') --stringparam SRCTIME $$(TZ= $(STAT) -f '%Sm' -t '%Y-%m-%dT%H:%M:%SZ' $(call quote,$(call uncompiled,$@))) --stringparam IDENTIFIER $(call quote,$(call localuri,$(call uncompiled,$@))) --stringparam PATH $(call quote,/$*) --stringparam CKSUM $$($(CKSUM) $(call quote,$(call uncompiled,$@)) | $(SED) 's/[ ].*//')$(if $(THISREV), --stringparam THISREV $(call quote,$(THISREV)),)$(if $(SRCREV), --stringparam SRCREV $(call quote,$(SRCREV)),) $(call quote,$(BUILDDIR)/transform.xslt) $(call quote,$<)
 $(call compiled,$(filter $(assetfiles),$(sourcefiles))) : $(BUILDDIR)/public/% : $$(call uncompiled,$$@)
 	@$(PRINTF) '%s\n' $(call quote,Compiling </$*>…)
 	$(silent)$(call ensuredirectory,$(dir $@))
diff --git a/README.markdown b/README.markdown
index 6b4f06e..f6a0864 100644
--- a/README.markdown
+++ b/README.markdown
@@ -379,6 +379,8 @@ Embeds are replaced with the parsed contents of a file, unless the file
   is an asset, in which case an `<html:object>` element is produced
   instead (with the contents of the asset file provided as a base64
   `data:` u·r·i).
+Embed replacements will be given a `@书社:identifier` attribute whose
+  value will match the `@xlink:href` of the embed.
 
 Embedding takes place after parsing but before transformation, so
   parsers are able to generate their own embeds.
@@ -401,14 +403,6 @@ Transforms are used to convert X·M·L files into their final output,
   after all necessary parsing and embedding has taken place.
 ⛩️📰 书社 comes with some transforms; namely :⁠—
 
-- **`transforms/attributes.xslt`:**
-  Applies transforms to the children of any `<书社:apply-attributes>`
-    elements, and then applies the attributes of the
-    `<书社:apply-attributes>` to each result child, replacing the
-    element with the result.
-  This is useful in combination with image embeds to apply alt‐text to
-    the resulting `<html:img>`.
-
 - **`transforms/asset.xslt`:**
   Converts `<html:object>` elements which correspond to recognized
     media types into the appropriate H·T·M·L elements, and deletes
@@ -423,8 +417,8 @@ Transforms are used to convert X·M·L files into their final output,
   - **`urn:fdc:ladys.computer:20231231:Shu1She4:title`:**
     Provides the title of the page.
 
-  ⛩️📰 书社 automatically encapsulates embeds so that their metadata
-    does not propogate up to the embedding document.
+  ⛩️📰 书社 automatically encapsulates H·T·M·L embeds so that their
+    metadata does not propogate up to the embedding document.
   To undo this behaviour, remove the `@itemscope` and `@itemtype`
     attributes from the embed during the transformation phase.
 
@@ -451,6 +445,10 @@ The following params are made available globally in parsers and
 - **`CKSUM`:**
   The checksum of the source file (⅌ `cksum`).
 
+- **`IDENTIFIER`:**
+  The ⛩️📰 书社 identifier of the source file (a u·r·i beginning with
+    `about:shushe`).
+
 - **`SRCREV`:**
   The value of the `SRCREV` variable (if present).
 
@@ -520,6 +518,29 @@ This mechanism can be used to allow transforms to insert content
 Output wrapping can be entirely disabled by adding a
   `@书社:disable-output-wrapping` attribute to the top‐level element in
   the result tree.
+This attribute will also prevent wrapping non‐H·T·M·L embeds with an
+  `<html:div>`.
+
+## Applying Attributes
+
+The `<书社:apply-attributes>` element will apply any attributes on the
+  element to the element(s) it wraps.
+It is especially useful in combination with embeds.
+
+The `<书社:apply-attributes-to-root>` element will apply any attributes
+  on the element to the root node of the final transformation result.
+It is especially useful in combination with output wrapping.
+
+In both cases, attributes from various sources are combined with
+  white·space between them.
+Attribute application takes place after all ordinary transforms have
+  completed.
+
+Both elements ignore attributes in the `xml:` namespace, except for
+  `@xml:lang`, which ignores all but the first definition (including
+  any already present on the root element).
+On H·T·M·L and S·V·G elements, `@lang` has the same behaviour as
+  `@xml:lang`.
 
 ## License
 
diff --git a/lib/catalog2parser.xslt b/lib/catalog2parser.xslt
index 4ea7bda..1faa880 100644
--- a/lib/catalog2parser.xslt
+++ b/lib/catalog2parser.xslt
@@ -24,6 +24,7 @@ If a copy of the M·P·L was not distributed with this file, You can obtain one
 		<xslt:transform version="1.0">
 			<xslt:param name="BUILDTIME" select="'1972-12-31T00:00:00Z'"/>
 			<xslt:param name="CKSUM" select="false"/>
+			<xslt:param name="IDENTIFIER" select="false"/>
 			<xslt:param name="SRCREV" select="false"/>
 			<xslt:param name="SRCTIME" select="'1972-12-31T00:00:00Z'"/>
 			<xslt:param name="THISREV" select="false"/>
@@ -94,7 +95,9 @@ If a copy of the M·P·L was not distributed with this file, You can obtain one
 									<xslt:attribute name="书社:parsed-by">
 										<xslt:value-of select="$id"/>
 										<xslt:if test="@书社:parsed-by">
-											<xslt:text> </xslt:text>
+											<xslt:text>
+												<text> </text>
+											</xslt:text>
 											<xslt:value-of select="@书社:parsed-by"/>
 										</xslt:if>
 									</xslt:attribute>
diff --git a/lib/catalog2transform.xslt b/lib/catalog2transform.xslt
index a6c83d6..b53e12a 100644
--- a/lib/catalog2transform.xslt
+++ b/lib/catalog2transform.xslt
@@ -9,6 +9,7 @@ If a copy of the M·P·L was not distributed with this file, You can obtain one
 -->
 <!DOCTYPE transform [
 	<!ENTITY 书社 "urn:fdc:ladys.computer:20231231:Shu1She4">
+	<!ENTITY xml "http://www.w3.org/XML/1998/namespace">
 ]>
 <transform
 	xmlns="http://www.w3.org/1999/XSL/Transform"
@@ -28,15 +29,17 @@ If a copy of the M·P·L was not distributed with this file, You can obtain one
 			<xslt:param name="BUILDTIME" select="'1972-12-31T00:00:00Z'"/>
 			<xslt:param name="CKSUM" select="false"/>
 			<xslt:param name="CATALOG" select="'catalog'"/>
+			<xslt:param name="IDENTIFIER" select="false"/>
 			<xslt:param name="PATH" select="'/unknown'"/>
 			<xslt:param name="SRCREV" select="false"/>
 			<xslt:param name="SRCTIME" select="'1972-12-31T00:00:00Z'"/>
 			<xslt:param name="THISREV" select="false"/>
+			<xslt:variable name="书社:source" select="/"/>
 			<xslt:variable name="书社:expansion">
-				<xslt:apply-templates select="/" mode="书社:expand"/>
+				<xslt:apply-templates select="$书社:source/node()" mode="书社:expand"/>
 			</xslt:variable>
 			<xslt:variable name="书社:result">
-				<xslt:apply-templates select="exsl:node-set($书社:expansion)/*"/>
+				<xslt:apply-templates select="exsl:node-set($书社:expansion)/node()"/>
 			</xslt:variable>
 			<for-each select="//catalog:uri">
 				<xslt:include href="{@uri}">
@@ -47,62 +50,290 @@ If a copy of the M·P·L was not distributed with this file, You can obtain one
 					</if>
 				</xslt:include>
 			</for-each>
-			<xslt:template match="/" priority="1">
-				<xslt:choose>
-					<xslt:when test="exsl:node-set($书社:result)/*[@书社:disable-output-wrapping]">
-						<xslt:for-each select="exsl:node-set($书社:result)/*">
+			<xslt:template name="书社:apply-attributes">
+				<xslt:param name="and-version" select="false"/>
+				<xslt:param name="context-nodes" select="/.."/>
+				<xslt:param name="destination-nodes" select="/.."/>
+				<xslt:variable name="additional-attributes" select="$context-nodes/@*"/>
+				<xslt:for-each select="$destination-nodes">
+					<xslt:variable name="existing-attributes" select="@*"/>
+					<xslt:choose>
+						<xslt:when test="self::*">
+							<xslt:variable name="context" select="."/>
+							<xslt:variable name="attribute-names">
+								<xslt:for-each select="$existing-attributes|$additional-attributes">
+									<书社:attribute>
+										<attribute name="local-name">
+											<text>{local-name()}</text>
+										</attribute>
+										<attribute name="name">
+											<text>{name()}</text>
+										</attribute>
+										<attribute name="namespace-uri">
+											<text>{namespace-uri()}</text>
+										</attribute>
+									</书社:attribute>
+								</xslt:for-each>
+							</xslt:variable>
+							<xslt:variable name="lang">
+								<xslt:choose>
+									<xslt:when test="(self::html:*|self::svg:*)/@lang">
+										<xslt:value-of select="@lang"/>
+									</xslt:when>
+									<xslt:when test="@xml:lang">
+										<xslt:value-of select="@xml:lang"/>
+									</xslt:when>
+									<xslt:when test="$additional-attributes[(parent::html:* or parent::svg:*) and namespace-uri()='' and local-name()='lang']">
+										<xslt:value-of select="$additional-attributes[(parent::html:* or parent::svg:*) and namespace-uri()='' and local-name()='lang']"/>
+									</xslt:when>
+									<xslt:when test="(self::html:* or self::svg:*) and $additional-attributes[(parent::书社:apply-attributes or parent::书社:apply-attributes-to-root) and namespace-uri()='' and local-name()='lang']">
+										<xslt:value-of select="$additional-attributes[(parent::书社:apply-attributes or parent::书社:apply-attributes-to-root) and namespace-uri()='' and local-name()='lang']"/>
+									</xslt:when>
+									<xslt:when test="$additional-attributes[namespace-uri()='&xml;' and local-name()='lang']">
+										<xslt:value-of select="$additional-attributes[namespace-uri()='&xml;' and local-name()='lang']"/>
+									</xslt:when>
+								</xslt:choose>
+							</xslt:variable>
 							<xslt:copy>
-								<xslt:if test="$THISREV">
+								<xslt:if test="$and-version and $THISREV">
 									<xslt:attribute name="书社:version">
 										<xslt:value-of select="$THISREV"/>
 									</xslt:attribute>
 								</xslt:if>
-								<xslt:copy-of select="@*[not(namespace-uri()='&书社;' and contains('disable-output-wrapping version', local-name()))]|node()"/>
+								<xslt:if test="string($lang)!=''">
+									<xslt:if test="self::html:* or self::svg:*">
+										<xslt:attribute name="lang">
+											<xslt:value-of select="$lang"/>
+										</xslt:attribute>
+									</xslt:if>
+									<xslt:attribute name="xml:lang">
+										<xslt:value-of select="$lang"/>
+									</xslt:attribute>
+								</xslt:if>
+								<xslt:for-each select="exsl:node-set($attribute-names)/*">
+									<xslt:choose>
+										<xslt:when test="@namespace-uri='&xml;'"/>
+										<xslt:when test="$context[self::html:* or self::svg:*] and @namespace-uri='' and @local-name='lang'"/>
+										<xslt:when test="@namespace-uri='&书社;' and (@local-name='destination' or @local-name='disable-output-wrapping' or @local-name='version')"/>
+										<xslt:when test="preceding-sibling::*[@local-name=current()/@local-name and @namespace-uri=current()/@namespace-uri]"/>
+										<xslt:otherwise>
+											<xslt:attribute>
+												<attribute name="name">
+													<text>{@name}</text>
+												</attribute>
+												<attribute name="namespace">
+													<text>{@namespace-uri}</text>
+												</attribute>
+												<xslt:for-each select="$existing-attributes[local-name()=current()/@local-name and namespace-uri()=current()/@namespace-uri]">
+													<xslt:value-of select="."/>
+													<xslt:if test="position()!=last()">
+														<xslt:text>
+															<text> </text>
+														</xslt:text>
+													</xslt:if>
+												</xslt:for-each>
+												<xslt:if test="$existing-attributes[local-name()=current()/@local-name and namespace-uri()=current()/@namespace-uri] and $additional-attributes[local-name()=current()/@local-name and namespace-uri()=current()/@namespace-uri]">
+													<xslt:text>
+														<text> </text>
+													</xslt:text>
+												</xslt:if>
+												<xslt:for-each select="$additional-attributes[local-name()=current()/@local-name and namespace-uri()=current()/@namespace-uri]">
+													<xslt:value-of select="."/>
+													<xslt:if test="position()!=last()">
+														<xslt:text>
+															<text> </text>
+														</xslt:text>
+													</xslt:if>
+												</xslt:for-each>
+											</xslt:attribute>
+										</xslt:otherwise>
+									</xslt:choose>
+								</xslt:for-each>
+								<xslt:copy-of select="node()"/>
 							</xslt:copy>
-						</xslt:for-each>
-					</xslt:when>
-					<xslt:otherwise>
-						<xslt:apply-templates select="exsl:node-set($书社:result)" mode="书社:wrap"/>
-					</xslt:otherwise>
-				</xslt:choose>
+						</xslt:when>
+						<xslt:otherwise>
+							<xslt:copy-of select="."/>
+						</xslt:otherwise>
+					</xslt:choose>
+				</xslt:for-each>
+			</xslt:template>
+			<xslt:template name="书社:wrap">
+				<xslt:param name="nodes" select="/.."/>
+				<xslt:variable name="modalinput">
+					<xslt:copy-of select="$nodes"/>
+					<xslt:copy-of select="document('')/xslt:transform/xslt:include"/>
+				</xslt:variable>
+				<xslt:variable name="metadata">
+					<xslt:copy-of select="$nodes[self::html:html]/html:head/node()|$nodes[self::html:head]/node()"/>
+					<xslt:apply-templates select="exsl:node-set($modalinput)/node()" mode="书社:metadata"/>
+				</xslt:variable>
+				<html:html>
+					<xslt:copy-of select="$nodes[self::html:html]/@*"/>
+					<xslt:if test="not($nodes[self::html:html]/@lang) and $nodes[self::html:*]/@lang|$nodes[self::svg:*]/@lang|$nodes/@xml:lang">
+						<xslt:attribute name="lang">
+							<xslt:value-of select="$nodes[self::html:*]/@lang|$nodes[self::svg:*]/@lang|$nodes/@xml:lang"/>
+						</xslt:attribute>
+					</xslt:if>
+					<xslt:if test="not($nodes[self::html:html]/@xml:lang) and $nodes[self::html:*]/@lang|$nodes[self::svg:*]/@lang|$nodes/@xml:lang">
+						<xslt:attribute name="xml:lang">
+							<xslt:value-of select="$nodes[self::html:*]/@lang|$nodes[self::svg:*]/@lang|$nodes/@xml:lang"/>
+						</xslt:attribute>
+					</xslt:if>
+					<html:head>
+						<xslt:copy-of select="$nodes[self::html:html]/html:head/@*|$nodes[self::html:head]/@*"/>
+						<html:title>
+							<xslt:for-each select="exsl:node-set($metadata)/html:title">
+								<xslt:value-of select="."/>
+								<xslt:if test="position()!=last()">
+									<xslt:text>
+										<text> </text>
+									</xslt:text>
+								</xslt:if>
+							</xslt:for-each>
+						</html:title>
+						<xslt:copy-of select="exsl:node-set($metadata)/node()[not(self::html:title)]"/>
+						<html:meta name="generator">
+							<xslt:attribute name="content">
+								<xslt:for-each select="exsl:node-set($metadata)/html:meta[@name='generator']">
+									<xslt:value-of select="@content"/>
+									<xslt:text>
+										<text>, </text>
+									</xslt:text>
+								</xslt:for-each>
+								<xslt:text>
+									<text>⛩️📰 书社</text>
+								</xslt:text>
+								<xslt:if test="$THISREV">
+									<xslt:text>
+										<text> (</text>
+									</xslt:text>
+									<xslt:value-of select="$THISREV"/>
+									<xslt:text>
+										<text>)</text>
+									</xslt:text>
+								</xslt:if>
+							</xslt:attribute>
+						</html:meta>
+					</html:head>
+					<html:body>
+						<xslt:copy-of select="$nodes[self::html:html]/html:body/@*|$nodes[self::html:body]/@*"/>
+						<xslt:apply-templates select="exsl:node-set($modalinput)/node()" mode="书社:header"/>
+						<xslt:copy-of select="$nodes[not(self::html:html or self::html:head or self::html:body)]|$nodes[self::html:html]/node()[not(self::html:head or self::html:body)]|$nodes[self::html:html]/html:body/node()|$nodes[self::html:body]/node()"/>
+						<xslt:apply-templates select="exsl:node-set($modalinput)/node()" mode="书社:footer"/>
+					</html:body>
+				</html:html>
+			</xslt:template>
+			<xslt:template match="/" priority="1">
+				<xslt:variable name="result-nodes" select="exsl:node-set($书社:result)/node()[not(self::书社:apply-attributes-to-root)]|exsl:node-set($书社:result)/书社:apply-attributes-to-root/descendant::node()[not(self::书社:apply-attributes-to-root) and not(ancestor::*[not(self::书社:apply-attributes-to-root)])]"/>
+				<xslt:variable name="root-with-attributes">
+					<xslt:choose>
+						<xslt:when test="$result-nodes/@书社:disable-output-wrapping|exsl:node-set($书社:result)//书社:apply-attributes-to-root/@书社:disable-output-wrapping">
+							<xslt:call-template name="书社:apply-attributes">
+								<xslt:with-param name="and-version" select="true"/>
+								<xslt:with-param name="context-nodes" select="exsl:node-set($书社:result)//书社:apply-attributes-to-root"/>
+								<xslt:with-param name="destination-nodes" select="$result-nodes"/>
+							</xslt:call-template>
+						</xslt:when>
+						<xslt:otherwise>
+							<xslt:variable name="wrapped-result">
+								<xslt:call-template name="书社:wrap">
+									<xslt:with-param name="nodes" select="$result-nodes"/>
+								</xslt:call-template>
+							</xslt:variable>
+							<xslt:call-template name="书社:apply-attributes">
+								<xslt:with-param name="and-version" select="true"/>
+								<xslt:with-param name="context-nodes" select="exsl:node-set($书社:result)//书社:apply-attributes-to-root"/>
+								<xslt:with-param name="destination-nodes" select="exsl:node-set($wrapped-result)/node()"/>
+							</xslt:call-template>
+						</xslt:otherwise>
+					</xslt:choose>
+				</xslt:variable>
+				<xslt:apply-templates select="exsl:node-set($root-with-attributes)/node()" mode="书社:application"/>
 			</xslt:template>
-			<xslt:template match="/*/*//@书社:disable-output-wrapping|//@书社:destination" priority="0"/>
 			<xslt:template match="@*|node()" priority="-1">
 				<xslt:copy>
 					<xslt:apply-templates select="@*|node()"/>
 				</xslt:copy>
 			</xslt:template>
-			<xslt:template match="/*" mode="书社:expand" priority="0">
+			<xslt:template match="@*|node()" mode="书社:application">
 				<xslt:copy>
-					<xslt:attribute name="itemscope">itemscope</xslt:attribute>
-					<xslt:attribute name="itemtype">
-						<xslt:text>&书社;:document</xslt:text>
-						<xslt:for-each select="exslstr:tokenize(@itemtype)/token">
-							<xslt:text> </xslt:text>
-							<xslt:value-of select="."/>
-						</xslt:for-each>
+					<xslt:apply-templates select="@*|node()" mode="书社:application"/>
+				</xslt:copy>
+			</xslt:template>
+			<xslt:template match="@书社:destination|@书社:disable-output-wrapping" mode="书社:application" priority="1"/>
+			<xslt:template match="书社:apply-attributes" mode="书社:application" priority="1">
+				<xslt:variable name="children">
+					<xslt:apply-templates select="node()" mode="书社:application"/>
+				</xslt:variable>
+				<xslt:call-template name="书社:apply-attributes">
+					<xslt:with-param name="context-nodes" select="."/>
+					<xslt:with-param name="destination-nodes" select="exsl:node-set($children)/node()"/>
+				</xslt:call-template>
+			</xslt:template>
+			<xslt:template match="书社:apply-attributes-to-root" mode="书社:application" priority="1">
+				<xslt:apply-templates select="node()" mode="书社:application"/>
+			</xslt:template>
+			<xslt:template match="/node()" mode="书社:expand" priority="0">
+				<xslt:copy>
+					<xslt:attribute name="书社:identifier">
+						<xslt:value-of select="$IDENTIFIER"/>
 					</xslt:attribute>
-					<xslt:apply-templates select="@*[not(namespace-uri()='' and (local-name()='itemscope' or local-name()='itemtype'))]|node()" mode="书社:expand"/>
+					<xslt:choose>
+						<xslt:when test="self::html:*">
+							<xslt:attribute name="itemscope">itemscope</xslt:attribute>
+							<xslt:attribute name="itemtype">
+								<xslt:text>
+									<text>&书社;:document</text>
+								</xslt:text>
+								<xslt:for-each select="exslstr:tokenize(@itemtype)/token">
+									<xslt:text>
+										<text> </text>
+									</xslt:text>
+									<xslt:value-of select="."/>
+								</xslt:for-each>
+							</xslt:attribute>
+							<xslt:apply-templates select="@*[not(namespace-uri()='' and (local-name()='itemscope' or local-name()='itemtype'))]|node()" mode="书社:expand"/>
+						</xslt:when>
+						<xslt:otherwise>
+							<xslt:apply-templates select="@*|node()" mode="书社:expand"/>
+						</xslt:otherwise>
+					</xslt:choose>
 				</xslt:copy>
 			</xslt:template>
 			<xslt:template match="书社:link[@xlink:show='embed']" mode="书社:expand" priority="1">
-				<xslt:variable name="uri" select="substring-before(document($CATALOG)//catalog:uri[@name=current()/@xlink:href]/@uri[1], '#')"/>
+				<xslt:variable name="identifier" select="string(@xlink:href)"/>
+				<xslt:variable name="uri" select="substring-before(document($CATALOG)//catalog:uri[@name=$identifier]/@uri[1], '#')"/>
 				<xslt:choose>
 					<xslt:when test="$uri">
 						<xslt:variable name="expanded">
 							<xslt:apply-templates select="document($uri)" mode="书社:expand"/>
 						</xslt:variable>
-						<xslt:for-each select="exsl:node-set($expanded)/*">
+						<xslt:for-each select="exsl:node-set($expanded)/node()">
 							<xslt:copy>
-								<xslt:attribute name="itemscope">itemscope</xslt:attribute>
-								<xslt:attribute name="itemtype">
-									<xslt:text>&书社;:embed</xslt:text>
-									<xslt:for-each select="exslstr:tokenize(@itemtype)/token[string()!='&书社;:document']">
-										<xslt:text> </xslt:text>
-										<xslt:value-of select="."/>
-									</xslt:for-each>
+								<xslt:attribute name="书社:identifier">
+									<xslt:value-of select="$identifier"/>
 								</xslt:attribute>
-								<xslt:copy-of select="@*[not(namespace-uri()='' and (local-name()='itemscope' or local-name()='itemtype'))]|node()"/>
+								<xslt:choose>
+									<xslt:when test="self::html:*">
+										<xslt:attribute name="itemscope">itemscope</xslt:attribute>
+										<xslt:attribute name="itemtype">
+											<xslt:text>
+												<text>&书社;:embed</text>
+											</xslt:text>
+											<xslt:for-each select="exslstr:tokenize(@itemtype)/token[string()!='&书社;:document']">
+												<xslt:text>
+													<text> </text>
+												</xslt:text>
+												<xslt:value-of select="."/>
+											</xslt:for-each>
+										</xslt:attribute>
+										<xslt:copy-of select="@*[not(namespace-uri()='' and (local-name()='itemscope' or local-name()='itemtype')) and not(namespace-uri()='&书社;' and local-name()='identifier')]|node()"/>
+									</xslt:when>
+									<xslt:otherwise>
+										<xslt:apply-templates select="@*[not(namespace-uri()='&书社;' and local-name()='identifier')]|node()" mode="书社:expand"/>
+									</xslt:otherwise>
+								</xslt:choose>
 							</xslt:copy>
 						</xslt:for-each>
 					</xslt:when>
@@ -113,7 +344,7 @@ If a copy of the M·P·L was not distributed with this file, You can obtain one
 					</xslt:otherwise>
 				</xslt:choose>
 			</xslt:template>
-			<xslt:template match="@*|text()|*[not(self::书社:link) or not(@xlink:show='embed')]" mode="书社:expand" priority="-1">
+			<xslt:template match="@*|node()[not(self::书社:link) or not(@xlink:show='embed')]" mode="书社:expand" priority="-1">
 				<xslt:copy>
 					<xslt:apply-templates select="@*|node()" mode="书社:expand"/>
 				</xslt:copy>
@@ -121,63 +352,6 @@ If a copy of the M·P·L was not distributed with this file, You can obtain one
 			<xslt:template match="text()" mode="书社:header"/>
 			<xslt:template match="text()" mode="书社:footer"/>
 			<xslt:template match="text()" mode="书社:metadata"/>
-			<xslt:template match="/" mode="书社:wrap">
-				<xslt:variable name="modalinput">
-					<xslt:copy-of select="node()"/>
-					<xslt:copy-of select="document('')/xslt:transform/xslt:include"/>
-				</xslt:variable>
-				<xslt:variable name="metadata">
-					<xslt:copy-of select="html:html/html:head/node()|html:head/node()"/>
-					<xslt:apply-templates select="exsl:node-set($modalinput)" mode="书社:metadata"/>
-				</xslt:variable>
-				<html:html>
-					<xslt:if test="$THISREV">
-						<xslt:attribute name="书社:version">
-							<xslt:value-of select="$THISREV"/>
-						</xslt:attribute>
-					</xslt:if>
-					<xslt:copy-of select="html:html/@*"/>
-					<xslt:if test="not(html:html/@lang) and (html:*/@lang|svg:*/@lang|*/@xml:lang)">
-						<xslt:attribute name="lang">
-							<xslt:value-of select="html:*/@lang|svg:*/@lang|*/@xml:lang"/>
-						</xslt:attribute>
-					</xslt:if>
-					<xslt:if test="not(html:html/@xml:lang) and (html:*/@lang|svg:*/@lang|*/@xml:lang)">
-						<xslt:attribute name="xml:lang">
-							<xslt:value-of select="html:*/@lang|svg:*/@lang|*/@xml:lang"/>
-						</xslt:attribute>
-					</xslt:if>
-					<html:head>
-						<xslt:copy-of select="html:html/html:head/@*|html:head/@*"/>
-						<html:title>
-							<xslt:for-each select="exsl:node-set($metadata)/html:title">
-								<xslt:value-of select="."/>
-							</xslt:for-each>
-						</html:title>
-						<xslt:copy-of select="exsl:node-set($metadata)/node()[not(self::*) or not(self::html:title)]"/>
-						<html:meta name="generator">
-							<xslt:attribute name="content">
-								<xslt:for-each select="exsl:node-set($metadata)/html:meta[@name='generator']">
-									<xslt:value-of select="@content"/>
-									<xslt:text>, </xslt:text>
-								</xslt:for-each>
-								<xslt:text>⛩️📰 书社</xslt:text>
-								<xslt:if test="$THISREV">
-									<xslt:text> (</xslt:text>
-									<xslt:value-of select="$THISREV"/>
-									<xslt:text>)</xslt:text>
-								</xslt:if>
-							</xslt:attribute>
-						</html:meta>
-					</html:head>
-					<html:body>
-						<xslt:copy-of select="html:html/html:body/@*|html:body/@*"/>
-						<xslt:apply-templates select="exsl:node-set($modalinput)" mode="书社:header"/>
-						<xslt:copy-of select="node()[not(self::html:html or self::html:head or self::html:body)]|html:html/node()[not(self::html:head or self::html:body)]|html:html/html:body/node()|html:body/node()"/>
-						<xslt:apply-templates select="exsl:node-set($modalinput)" mode="书社:footer"/>
-					</html:body>
-				</html:html>
-			</xslt:template>
 			<xslt:output method="xml" encoding="UTF-8" cdata-section-elements="html:script html:style html:textarea"/>
 		</xslt:transform>
 	</template>
diff --git a/transforms/asset.xslt b/transforms/asset.xslt
index 0242f43..55b449b 100644
--- a/transforms/asset.xslt
+++ b/transforms/asset.xslt
@@ -19,20 +19,30 @@ If a copy of the M·P·L was not distributed with this file, You can obtain one
 	<书社:id>urn:fdc:ladys.computer:20231231:Shu1She4:asset.xslt</书社:id>
 	<template match="html:style|html:object[@type='text/css']"/>
 	<template match="html:object[@type='text/javascript']">
-		<html:script type="{@type}" src="{@data}"/>
+		<html:script type="{@type}" src="{@data}">
+			<copy-of select="@书社:identifier"/>
+		</html:script>
 	</template>
 	<template match="html:object[starts-with(@type, 'audio/')]">
-		<html:audio controls="" src="{@data}"/>
+		<html:audio controls="" src="{@data}">
+			<copy-of select="@书社:identifier"/>
+		</html:audio>
 	</template>
 	<template match="html:object[starts-with(@type, 'image/') and not(@type='image/svg+xml')]">
-		<html:img src="{@data}"/>
+		<html:img src="{@data}">
+			<copy-of select="@书社:identifier"/>
+		</html:img>
 	</template>
 	<template match="html:object[starts-with(@type, 'video/')]">
-		<html:video controls="" src="{@data}"/>
+		<html:video controls="" src="{@data}">
+			<copy-of select="@书社:identifier"/>
+		</html:video>
 	</template>
 	<template match="xslt:include[@书社:id='urn:fdc:ladys.computer:20231231:Shu1She4:asset.xslt']" mode="书社:metadata">
 		<for-each select="exsl:node-set($书社:expansion)//html:object[@type='text/css']">
-			<html:link rel="stylesheet" type="text/css" href="{@data}"/>
+			<html:link rel="stylesheet" type="text/css" href="{@data}">
+				<copy-of select="@书社:identifier"/>
+			</html:link>
 		</for-each>
 		<for-each select="exsl:node-set($书社:expansion)//html:style">
 			<copy-of select="."/>
diff --git a/transforms/attributes.xslt b/transforms/attributes.xslt
deleted file mode 100644
index e91e7cb..0000000
--- a/transforms/attributes.xslt
+++ /dev/null
@@ -1,41 +0,0 @@
-<?xml version="1.0"?>
-<!--
-⁌ ⛩️📰 书社 ∷ transforms/attributes.xslt
-
-© 2024 Lady [@ Lady’s Computer]
-
-This Source Code Form is subject to the terms of the Mozilla Public License, v 2.0.
-If a copy of the M·P·L was not distributed with this file, You can obtain one at <https://mozilla.org/MPL/2.0/>.
--->
-<transform
-	xmlns="http://www.w3.org/1999/XSL/Transform"
-	xmlns:exsl="http://exslt.org/common"
-	xmlns:书社="urn:fdc:ladys.computer:20231231:Shu1She4"
-	exclude-result-prefixes="exsl"
-	version="1.0"
->
-	<书社:id>urn:fdc:ladys.computer:20231231:Shu1She4:attributes.xslt</书社:id>
-	<template match="书社:apply-attributes">
-		<for-each select="node()">
-			<choose>
-				<when test="self::*">
-					<variable name="original" select="."/>
-					<variable name="result">
-						<apply-templates select="."/>
-					</variable>
-					<for-each select="exsl:node-set($result)/*">
-						<copy>
-							<for-each select="@*|$original/../@*">
-								<copy/>
-							</for-each>
-							<apply-templates/>
-						</copy>
-					</for-each>
-				</when>
-				<otherwise>
-					<apply-templates select="."/>
-				</otherwise>
-			</choose>
-		</for-each>
-	</template>
-</transform>