Skip to content

Content of file DefaultSitemapViewController.java.html

<?xml version="1.0" encoding="UTF-8"?><!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Strict//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd"><html xmlns="http://www.w3.org/1999/xhtml" lang="en"><head><meta http-equiv="Content-Type" content="text/html;charset=UTF-8"/><link rel="stylesheet" href="../../jacoco-resources/report.css" type="text/css"/><link rel="shortcut icon" href="../../jacoco-resources/report.gif" type="image/gif"/><title>DefaultSitemapViewController.java</title><link rel="stylesheet" href="../../jacoco-resources/prettify.css" type="text/css"/><script type="text/javascript" src="../../jacoco-resources/prettify.js"></script></head><body onload="window['PR_TAB_WIDTH']=4;prettyPrint()"><div class="breadcrumb" id="breadcrumb"><span class="info"><a href="../../jacoco-sessions.html" class="el_session">Sessions</a></span><a href="../../index.html" class="el_report">NorthernWind :: Frontend :: Components :: HTML Template</a> &gt; <a href="../index.html" class="el_bundle">it-tidalwave-northernwind-frontend-components</a> &gt; <a href="index.source.html" class="el_package">it.tidalwave.northernwind.frontend.ui.component.sitemap</a> &gt; <span class="el_source">DefaultSitemapViewController.java</span></div><h1>DefaultSitemapViewController.java</h1><pre class="source lang-java linenums">/*
 * #%L
 * *********************************************************************************************************************
 *
 * NorthernWind - lightweight CMS
 * http://northernwind.tidalwave.it - git clone https://bitbucket.org/tidalwave/northernwind-src.git
 * %%
 * Copyright (C) 2011 - 2023 Tidalwave s.a.s. (http://tidalwave.it)
 * %%
 * *********************************************************************************************************************
 *
 * Licensed under the Apache License, Version 2.0 (the &quot;License&quot;); you may not use this file except in compliance with
 * the License. You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on
 * an &quot;AS IS&quot; BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.  See the License for the
 * specific language governing permissions and limitations under the License.
 *
 * *********************************************************************************************************************
 *
 *
 * *********************************************************************************************************************
 * #L%
 */
package it.tidalwave.northernwind.frontend.ui.component.sitemap;

import javax.annotation.CheckForNull;
import javax.annotation.Nonnull;
import java.time.ZonedDateTime;
import java.util.Optional;
import java.util.Set;
import java.util.SortedSet;
import java.util.TreeSet;
import it.tidalwave.northernwind.core.model.HttpStatusException;
import it.tidalwave.northernwind.core.model.ResourceProperties;
import it.tidalwave.northernwind.core.model.Site;
import it.tidalwave.northernwind.core.model.SiteNode;
import it.tidalwave.northernwind.frontend.ui.Layout;
import it.tidalwave.northernwind.frontend.ui.RenderContext;
import lombok.EqualsAndHashCode;
import lombok.Getter;
import lombok.RequiredArgsConstructor;
import lombok.ToString;
import lombok.extern.slf4j.Slf4j;
import static java.util.stream.Collectors.*;
import static it.tidalwave.northernwind.core.model.Content.*;
import static it.tidalwave.northernwind.core.model.SiteNode._SiteNode_;
import static it.tidalwave.northernwind.frontend.ui.component.blog.DefaultBlogViewController.TIME0;

/***********************************************************************************************************************
 *
 * &lt;p&gt;A default implementation of the {@link SitemapViewController} that is independent of the presentation technology.
 * This class is capable to render the sitemap of a {@link Site}.&lt;/p&gt;
 *
 * &lt;p&gt;Supported properties of any {@link SiteNode} in the site:&lt;/p&gt;
 *
 * &lt;ul&gt;
 * &lt;li&gt;{@code P_SITEMAP_PRIORITY}: the priority of the {@code SiteNode} - if zero, the node is ignored;&lt;/li&gt;
 * &lt;li&gt;{@code P_SITEMAP_CHILDREN_PRIORITY}: same as {@code P_SITEMAP_PRIORITY}, but for child nodes;&lt;/li&gt;
 * &lt;li&gt;{@code P_LATEST_MODIFICATION_DATE}: the date-time of the latest modification;&lt;/li&gt;
 * &lt;li&gt;{@code P_SITEMAP_CHANGE_FREQUENCY}: the supposed change frequency of the {@code SiteNode}.&lt;/li&gt;
 * &lt;/ul&gt;
 *
 * &lt;p&gt;Concrete implementations must provide one method for rendering the calendar:&lt;/p&gt;
 *
 * &lt;ul&gt;
 * &lt;li&gt;{@link #render(java.util.List)}&lt;/li&gt;
 * &lt;/ul&gt;
 *
 * @author  Fabrizio Giudici
 *
 **********************************************************************************************************************/
<span class="fc" id="L75">@RequiredArgsConstructor @Slf4j</span>
public abstract class DefaultSitemapViewController implements SitemapViewController
  {
<span class="pc bpc" id="L78" title="19 of 32 branches missed.">    @RequiredArgsConstructor @ToString @Getter @EqualsAndHashCode</span>
    protected static class Entry implements Comparable&lt;Entry&gt;
      {
        @Nonnull
<span class="fc" id="L82">        private final String location;</span>

        @Nonnull
<span class="fc" id="L85">        private final ZonedDateTime lastModification;</span>

        @Nonnull
<span class="fc" id="L88">        private final String changeFrequency;</span>

<span class="fc" id="L90">        private final float priority;</span>

        @Override
        public int compareTo (@Nonnull final Entry other)
          {
<span class="fc bfc" id="L95" title="All 2 branches covered.">            return this.equals(other) ? 0 : this.location.compareTo(other.location);</span>
          }
      }

    @Nonnull
    private final SiteNode siteNode;

    @Nonnull
    private final SitemapView view;

    /*******************************************************************************************************************
     *
     * {@inheritDoc}
     *
     ******************************************************************************************************************/
    @Override
    public void renderView (@Nonnull final RenderContext context)
      {
<span class="fc" id="L113">        final SortedSet&lt;Entry&gt; entries = new TreeSet&lt;&gt;();</span>

<span class="fc" id="L115">        siteNode.getSite().find(_SiteNode_).stream().forEach(node -&gt;</span>
          {
<span class="fc" id="L117">            final var layout = node.getLayout();</span>

            // Prevents infinite recursion
<span class="pc bpc" id="L120" title="1 of 2 branches missed.">            if (!layout.getTypeUri().startsWith(&quot;http://northernwind.tidalwave.it/component/Sitemap/&quot;))</span>
              {
                // FIXME: should probably skip children of sitenodes with managePathParams
                // FIXME: for instance, Calendar would benefit
                // FIXME: Would Blog benefit? It should, as it manages its own children
                // FIXME: Places and Themes should move managePathParams=true to each children
                // FIXME: Problem, the root gallery needs managePathParams=true to load images.xml
<span class="fc" id="L127">                log.debug(&quot;&gt;&gt;&gt;&gt; sitemap processing {} / layout {} ...&quot;, node.getRelativeUri(), layout);</span>

<span class="fc" id="L129">                newEntry(node, null).ifPresent(entries::add);</span>

<span class="fc" id="L131">                layout.accept(new Visitor&lt;Layout, Void&gt;()</span>
<span class="fc" id="L132">                  {</span>
                    @Override
                    public void visit (@Nonnull final Layout childLayout)
                      {
                        try
                          {
<span class="fc" id="L138">                            entries.addAll(childLayout.createViewAndController(node).getController()</span>
<span class="fc" id="L139">                                .findVirtualSiteNodes()</span>
<span class="fc" id="L140">                                .results()</span>
<span class="fc" id="L141">                                .stream()</span>
<span class="fc" id="L142">                                .peek(e -&gt; log.debug(&quot;&gt;&gt;&gt;&gt;&gt;&gt;&gt;&gt; added virtual node: {}&quot;, e.getRelativeUri()))</span>
<span class="fc" id="L143">                                .flatMap(childNode -&gt; newEntry(node, childNode).stream())</span>
<span class="fc" id="L144">                                .collect(toList()));</span>
                          }
<span class="nc" id="L146">                        catch (HttpStatusException e)</span>
                          {
<span class="nc" id="L148">                            log.warn(&quot;sitemap for {} threw {}&quot;, node.getRelativeUri(), e.toString());</span>
                          }
<span class="nc" id="L150">                        catch (Exception e)</span>
                          {
<span class="nc" id="L152">                            log.warn(&quot;Skipped item because of {} - root cause {}&quot;, e, rootCause(e).toString());</span>
<span class="pc" id="L153">                          }</span>
<span class="fc" id="L154">                      }</span>
                  });
              }
<span class="fc" id="L157">          });</span>

<span class="fc" id="L159">        render(entries);</span>
<span class="fc" id="L160">      }</span>

    /*******************************************************************************************************************
     *
     *
     *
     ******************************************************************************************************************/
    protected abstract void render (@Nonnull Set&lt;? extends Entry&gt; entries);

    /*******************************************************************************************************************
     *
     *
     ******************************************************************************************************************/
    @Nonnull
    protected final ResourceProperties getViewProperties()
      {
<span class="fc" id="L176">        return siteNode.getPropertyGroup(view.getId());</span>
      }

    /*******************************************************************************************************************
     *
     *
     ******************************************************************************************************************/
    @Nonnull
    private static Optional&lt;Entry&gt; newEntry (@Nonnull final SiteNode siteNode,
                                             @CheckForNull final SiteNode childSiteNode)
      {
<span class="fc bfc" id="L187" title="All 2 branches covered.">        final var node = (childSiteNode != null) ? childSiteNode : siteNode;</span>
<span class="fc" id="L188">        final var properties = node.getProperties();</span>
        //
        // FIXME: if you put the sitemap property straightly into the child site node, you can simplify a lot,
if you put the sitemap property straightly into the child site node, you can simplify a lot,
// just using a single property and only peeking into a single node <span class="fc bfc" id="L192" title="All 2 branches covered."> final var priorityKey = (childSiteNode == null) ? P_SITEMAP_PRIORITY : P_SITEMAP_CHILDREN_PRIORITY;</span> <span class="fc" id="L193"> final float sitemapPriority = siteNode.getProperty(priorityKey).orElse(0.5f);</span> <span class="fc bfc" id="L195" title="All 2 branches covered."> return (sitemapPriority == 0)</span> <span class="fc" id="L196"> ? Optional.empty()</span> <span class="fc" id="L197"> : Optional.of(new Entry(siteNode.getSite().createLink(node.getRelativeUri()),</span> <span class="fc" id="L198"> properties.getProperty(P_LATEST_MODIFICATION_DATE).orElse(TIME0),</span> <span class="fc" id="L199"> properties.getProperty(P_SITEMAP_CHANGE_FREQUENCY).orElse(&quot;daily&quot;),</span> sitemapPriority)); } /******************************************************************************************************************* * * ******************************************************************************************************************/ @Nonnull private static Throwable rootCause (@Nonnull final Throwable t) { <span class="nc" id="L210"> final var cause = t.getCause();</span> <span class="nc bnc" id="L211" title="All 2 branches missed."> return (cause != null) ? rootCause(cause) : t;</span> } } </pre><div class="footer"><span class="right">Created with <a href="http://www.jacoco.org/jacoco">JaCoCo</a> 0.8.9.202303310957</span></div></body></html>