feat(front-end): sitemap update
This commit is contained in:
@@ -0,0 +1,37 @@
|
||||
package com.printcalculator.controller;
|
||||
|
||||
import com.printcalculator.service.shop.ShopSitemapService;
|
||||
import org.springframework.beans.factory.annotation.Value;
|
||||
import org.springframework.http.CacheControl;
|
||||
import org.springframework.http.MediaType;
|
||||
import org.springframework.http.ResponseEntity;
|
||||
import org.springframework.web.bind.annotation.GetMapping;
|
||||
import org.springframework.web.bind.annotation.RestController;
|
||||
|
||||
import java.time.Duration;
|
||||
|
||||
@RestController
|
||||
public class SitemapController {
|
||||
private final ShopSitemapService shopSitemapService;
|
||||
private final long cacheSeconds;
|
||||
|
||||
public SitemapController(
|
||||
ShopSitemapService shopSitemapService,
|
||||
@Value("${app.sitemap.shop.cache-seconds:3600}") long cacheSeconds
|
||||
) {
|
||||
this.shopSitemapService = shopSitemapService;
|
||||
this.cacheSeconds = Math.max(cacheSeconds, 0L);
|
||||
}
|
||||
|
||||
@GetMapping(value = "/api/sitemap-shop.xml", produces = MediaType.APPLICATION_XML_VALUE)
|
||||
public ResponseEntity<String> getShopSitemap() {
|
||||
CacheControl cacheControl = cacheSeconds > 0
|
||||
? CacheControl.maxAge(Duration.ofSeconds(cacheSeconds)).cachePublic()
|
||||
: CacheControl.noCache();
|
||||
|
||||
return ResponseEntity.ok()
|
||||
.contentType(MediaType.parseMediaType("application/xml;charset=UTF-8"))
|
||||
.cacheControl(cacheControl)
|
||||
.body(shopSitemapService.getShopSitemapXml());
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,242 @@
|
||||
package com.printcalculator.service.shop;
|
||||
|
||||
import com.printcalculator.entity.ShopCategory;
|
||||
import com.printcalculator.entity.ShopProduct;
|
||||
import com.printcalculator.repository.ShopCategoryRepository;
|
||||
import com.printcalculator.repository.ShopProductRepository;
|
||||
import org.springframework.beans.factory.annotation.Value;
|
||||
import org.springframework.stereotype.Service;
|
||||
import org.springframework.transaction.annotation.Transactional;
|
||||
|
||||
import java.net.URLEncoder;
|
||||
import java.nio.charset.StandardCharsets;
|
||||
import java.text.Normalizer;
|
||||
import java.time.Clock;
|
||||
import java.time.Duration;
|
||||
import java.time.Instant;
|
||||
import java.time.OffsetDateTime;
|
||||
import java.time.format.DateTimeFormatter;
|
||||
import java.util.LinkedHashMap;
|
||||
import java.util.List;
|
||||
import java.util.Locale;
|
||||
import java.util.Map;
|
||||
import java.util.Set;
|
||||
import java.util.UUID;
|
||||
import java.util.stream.Collectors;
|
||||
|
||||
@Service
|
||||
@Transactional(readOnly = true)
|
||||
public class ShopSitemapService {
|
||||
private static final List<String> SUPPORTED_LANGUAGES = ShopProduct.SUPPORTED_LANGUAGES;
|
||||
private static final String DEFAULT_LANGUAGE = "it";
|
||||
private static final DateTimeFormatter LASTMOD_FORMATTER = DateTimeFormatter.ISO_OFFSET_DATE_TIME;
|
||||
|
||||
private final ShopCategoryRepository shopCategoryRepository;
|
||||
private final ShopProductRepository shopProductRepository;
|
||||
private final String frontendBaseUrl;
|
||||
private final Duration cacheTtl;
|
||||
private final Clock clock;
|
||||
|
||||
private volatile CachedSitemap cachedSitemap;
|
||||
|
||||
public ShopSitemapService(ShopCategoryRepository shopCategoryRepository,
|
||||
ShopProductRepository shopProductRepository,
|
||||
@Value("${app.frontend.base-url:http://localhost:4200}") String frontendBaseUrl,
|
||||
@Value("${app.sitemap.shop.cache-seconds:900}") long cacheSeconds) {
|
||||
this(shopCategoryRepository, shopProductRepository, frontendBaseUrl, cacheSeconds, Clock.systemUTC());
|
||||
}
|
||||
|
||||
ShopSitemapService(ShopCategoryRepository shopCategoryRepository,
|
||||
ShopProductRepository shopProductRepository,
|
||||
String frontendBaseUrl,
|
||||
long cacheSeconds,
|
||||
Clock clock) {
|
||||
this.shopCategoryRepository = shopCategoryRepository;
|
||||
this.shopProductRepository = shopProductRepository;
|
||||
this.frontendBaseUrl = normalizeBaseUrl(frontendBaseUrl);
|
||||
this.cacheTtl = cacheSeconds > 0 ? Duration.ofSeconds(cacheSeconds) : Duration.ZERO;
|
||||
this.clock = clock;
|
||||
}
|
||||
|
||||
public String getShopSitemapXml() {
|
||||
Instant now = Instant.now(clock);
|
||||
CachedSitemap current = cachedSitemap;
|
||||
if (current != null && now.isBefore(current.expiresAt())) {
|
||||
return current.xml();
|
||||
}
|
||||
|
||||
synchronized (this) {
|
||||
current = cachedSitemap;
|
||||
now = Instant.now(clock);
|
||||
if (current != null && now.isBefore(current.expiresAt())) {
|
||||
return current.xml();
|
||||
}
|
||||
|
||||
String xml = buildSitemapXml();
|
||||
Instant expiresAt = cacheTtl.isZero() ? now : now.plus(cacheTtl);
|
||||
cachedSitemap = new CachedSitemap(xml, expiresAt);
|
||||
return xml;
|
||||
}
|
||||
}
|
||||
|
||||
private String buildSitemapXml() {
|
||||
List<ShopCategory> activeCategories = shopCategoryRepository.findAllByIsActiveTrueOrderBySortOrderAscNameAsc();
|
||||
Set<UUID> activeCategoryIds = activeCategories.stream()
|
||||
.map(ShopCategory::getId)
|
||||
.collect(Collectors.toSet());
|
||||
|
||||
List<ShopProduct> activeProducts = shopProductRepository.findAllByIsActiveTrueOrderByIsFeaturedDescSortOrderAscNameAsc();
|
||||
|
||||
StringBuilder xml = new StringBuilder(16_384);
|
||||
xml.append("<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n");
|
||||
xml.append("<urlset xmlns=\"http://www.sitemaps.org/schemas/sitemap/0.9\" ");
|
||||
xml.append("xmlns:xhtml=\"http://www.w3.org/1999/xhtml\">\n");
|
||||
|
||||
appendCategoryUrls(xml, activeCategories);
|
||||
appendProductUrls(xml, activeProducts, activeCategoryIds);
|
||||
|
||||
xml.append("</urlset>\n");
|
||||
return xml.toString();
|
||||
}
|
||||
|
||||
private void appendCategoryUrls(StringBuilder xml, List<ShopCategory> categories) {
|
||||
for (ShopCategory category : categories) {
|
||||
if (!Boolean.TRUE.equals(category.getIndexable())) {
|
||||
continue;
|
||||
}
|
||||
|
||||
String encodedSlug = pathEncodeSegment(category.getSlug());
|
||||
Map<String, String> hrefByLanguage = new LinkedHashMap<>();
|
||||
for (String language : SUPPORTED_LANGUAGES) {
|
||||
hrefByLanguage.put(language, frontendBaseUrl + "/" + language + "/shop/" + encodedSlug);
|
||||
}
|
||||
|
||||
appendUrlEntry(xml, hrefByLanguage, category.getUpdatedAt());
|
||||
}
|
||||
}
|
||||
|
||||
private void appendProductUrls(StringBuilder xml,
|
||||
List<ShopProduct> products,
|
||||
Set<UUID> activeCategoryIds) {
|
||||
for (ShopProduct product : products) {
|
||||
if (!Boolean.TRUE.equals(product.getIndexable())) {
|
||||
continue;
|
||||
}
|
||||
if (product.getCategory() == null || !activeCategoryIds.contains(product.getCategory().getId())) {
|
||||
continue;
|
||||
}
|
||||
|
||||
Map<String, String> hrefByLanguage = new LinkedHashMap<>();
|
||||
for (String language : SUPPORTED_LANGUAGES) {
|
||||
String publicSegment = localizedProductPathSegment(product, language);
|
||||
hrefByLanguage.put(language, frontendBaseUrl + "/" + language + "/shop/p/" + pathEncodeSegment(publicSegment));
|
||||
}
|
||||
|
||||
appendUrlEntry(xml, hrefByLanguage, product.getUpdatedAt());
|
||||
}
|
||||
}
|
||||
|
||||
private void appendUrlEntry(StringBuilder xml,
|
||||
Map<String, String> hrefByLanguage,
|
||||
OffsetDateTime lastmod) {
|
||||
String defaultHref = hrefByLanguage.get(DEFAULT_LANGUAGE);
|
||||
if (defaultHref == null || defaultHref.isBlank()) {
|
||||
return;
|
||||
}
|
||||
|
||||
xml.append(" <url>\n");
|
||||
xml.append(" <loc>").append(xmlEscape(defaultHref)).append("</loc>\n");
|
||||
|
||||
for (String language : SUPPORTED_LANGUAGES) {
|
||||
String href = hrefByLanguage.get(language);
|
||||
if (href == null || href.isBlank()) {
|
||||
continue;
|
||||
}
|
||||
xml.append(" <xhtml:link rel=\"alternate\" hreflang=\"")
|
||||
.append(language)
|
||||
.append("\" href=\"")
|
||||
.append(xmlEscape(href))
|
||||
.append("\" />\n");
|
||||
}
|
||||
|
||||
xml.append(" <xhtml:link rel=\"alternate\" hreflang=\"x-default\" href=\"")
|
||||
.append(xmlEscape(defaultHref))
|
||||
.append("\" />\n");
|
||||
|
||||
if (lastmod != null) {
|
||||
xml.append(" <lastmod>").append(LASTMOD_FORMATTER.format(lastmod)).append("</lastmod>\n");
|
||||
}
|
||||
|
||||
xml.append(" </url>\n");
|
||||
}
|
||||
|
||||
private String localizedProductPathSegment(ShopProduct product, String language) {
|
||||
String localizedName = product.getNameForLanguage(language);
|
||||
String idPrefix = productIdPrefix(product.getId());
|
||||
String tail = firstNonBlank(slugify(localizedName), slugify(product.getSlug()), "product");
|
||||
return idPrefix.isBlank() ? tail : idPrefix + "-" + tail;
|
||||
}
|
||||
|
||||
private String productIdPrefix(UUID productId) {
|
||||
if (productId == null) {
|
||||
return "";
|
||||
}
|
||||
String raw = productId.toString().trim().toLowerCase(Locale.ROOT);
|
||||
int dashIndex = raw.indexOf('-');
|
||||
if (dashIndex > 0) {
|
||||
return raw.substring(0, dashIndex);
|
||||
}
|
||||
return raw.length() >= 8 ? raw.substring(0, 8) : raw;
|
||||
}
|
||||
|
||||
static String slugify(String rawValue) {
|
||||
String safeValue = rawValue == null ? "" : rawValue;
|
||||
String normalized = Normalizer.normalize(safeValue, Normalizer.Form.NFD)
|
||||
.replaceAll("\\p{M}+", "")
|
||||
.toLowerCase(Locale.ROOT)
|
||||
.replaceAll("[^a-z0-9]+", "-")
|
||||
.replaceAll("^-+|-+$", "")
|
||||
.replaceAll("-{2,}", "-");
|
||||
return normalized;
|
||||
}
|
||||
|
||||
private String firstNonBlank(String... values) {
|
||||
if (values == null) {
|
||||
return null;
|
||||
}
|
||||
for (String value : values) {
|
||||
if (value != null && !value.isBlank()) {
|
||||
return value;
|
||||
}
|
||||
}
|
||||
return null;
|
||||
}
|
||||
|
||||
private String pathEncodeSegment(String rawSegment) {
|
||||
String safeSegment = rawSegment == null ? "" : rawSegment;
|
||||
return URLEncoder.encode(safeSegment, StandardCharsets.UTF_8).replace("+", "%20");
|
||||
}
|
||||
|
||||
private String xmlEscape(String value) {
|
||||
return String.valueOf(value)
|
||||
.replace("&", "&")
|
||||
.replace("<", "<")
|
||||
.replace(">", ">")
|
||||
.replace("\"", """)
|
||||
.replace("'", "'");
|
||||
}
|
||||
|
||||
private String normalizeBaseUrl(String baseUrl) {
|
||||
String normalized = (baseUrl == null ? "" : baseUrl).trim();
|
||||
if (normalized.isBlank()) {
|
||||
return "http://localhost:4200";
|
||||
}
|
||||
while (normalized.endsWith("/")) {
|
||||
normalized = normalized.substring(0, normalized.length() - 1);
|
||||
}
|
||||
return normalized;
|
||||
}
|
||||
|
||||
private record CachedSitemap(String xml, Instant expiresAt) {
|
||||
}
|
||||
}
|
||||
@@ -56,6 +56,7 @@ app.mail.contact-request.admin.enabled=${APP_MAIL_CONTACT_REQUEST_ADMIN_ENABLED:
|
||||
app.mail.contact-request.admin.address=${APP_MAIL_CONTACT_REQUEST_ADMIN_ADDRESS:info@3d-fab.ch}
|
||||
app.mail.contact-request.customer.enabled=${APP_MAIL_CONTACT_REQUEST_CUSTOMER_ENABLED:true}
|
||||
app.frontend.base-url=${APP_FRONTEND_BASE_URL:http://localhost:4200}
|
||||
app.sitemap.shop.cache-seconds=${APP_SITEMAP_SHOP_CACHE_SECONDS:3600}
|
||||
|
||||
# Admin back-office authentication
|
||||
admin.password=${ADMIN_PASSWORD}
|
||||
|
||||
Reference in New Issue
Block a user