All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.composum.ai.backend.slingbase.experimential.AITemplatingServlet Maven / Gradle / Ivy

Go to download

Common Functionality for Composum AI specific to Sling but would be useable in both Composum and AEM and similar.

The newest version!
package com.composum.ai.backend.slingbase.experimential;

import java.io.IOException;
import java.net.URI;
import java.util.ArrayList;
import java.util.Iterator;
import java.util.List;
import java.util.stream.Collectors;
import java.util.stream.Stream;

import javax.servlet.Servlet;
import javax.servlet.ServletException;

import org.apache.commons.lang3.StringUtils;
import org.apache.sling.api.SlingHttpServletRequest;
import org.apache.sling.api.SlingHttpServletResponse;
import org.apache.sling.api.resource.Resource;
import org.apache.sling.api.servlets.HttpConstants;
import org.apache.sling.api.servlets.ServletResolverConstants;
import org.apache.sling.api.servlets.SlingAllMethodsServlet;
import org.osgi.framework.Constants;
import org.osgi.service.component.annotations.Component;
import org.osgi.service.component.annotations.Reference;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import com.composum.ai.backend.slingbase.ApproximateMarkdownService;
import com.google.gson.Gson;
import com.google.gson.GsonBuilder;
import com.google.gson.stream.JsonWriter;

/**
 * Servlet providing services related to templating.
 * Exports the methods of {@link AITemplatingService} via a servlet.
 * URL is /bin/cpm/ai/experimental/templating.{methodname}.json
 * (Apache Sling conform), POST only, returns JSON,
 * currently only a string message telling whether it was successful. All parameters of the methods are passed
 * HTML form style (application/x-www-form-urlencoded).
 *
 * @see "10PageTemplating.md"
 */
@Component(service = Servlet.class,
        property = {
                Constants.SERVICE_DESCRIPTION + "=Composum AI Templating Servlet (experimental)",
                ServletResolverConstants.SLING_SERVLET_PATHS + "=/bin/cpm/ai/experimental/templating",
                ServletResolverConstants.SLING_SERVLET_METHODS + "=" + HttpConstants.METHOD_POST
        })
public class AITemplatingServlet extends SlingAllMethodsServlet {

    private static final Logger LOG = LoggerFactory.getLogger(AITemplatingServlet.class);

    /**
     * Parameter that gives the page to transform.
     */
    public static final String PARAM_RESOURCE_PATH = "resourcePath";
    /**
     * Parameter that gives an additional prompt to add to the AI request.
     */
    public static final String PARAM_ADDITIONAL_PROMPT = "additionalPrompt";
    /**
     * Parameter that gives additional URLs or paths to pages with background information to provide data to the AI.
     */
    public static final String PARAM_ADDITIONAL_URLS = "additionalUrls";
    /**
     * Parameter that gives additional background information to provide data to the AI (not a prompt!).
     */
    public static final String PARAM_BACKGROUND_INFORMATION = "backgroundInformation";

    @Reference
    private AITemplatingService aiTemplatingService;

    @Reference
    private ApproximateMarkdownService approximateMarkdownService;

    protected final Gson gson = new GsonBuilder().disableHtmlEscaping().create();

    protected enum Method {
        replacePromptsInResource, resetToPrompts
    }

    @Override
    protected void doPost(SlingHttpServletRequest request, SlingHttpServletResponse response) throws ServletException, IOException {
        String method = request.getRequestPathInfo().getSelectorString();
        switch (Method.valueOf(method)) {
            case replacePromptsInResource:
                replacePromptsInResource(request, response);
                break;
            case resetToPrompts:
                resetToPrompts(request, response);
                break;
            default:
                response.setStatus(404);
        }
    }

    protected void replacePromptsInResource(SlingHttpServletRequest request, SlingHttpServletResponse response) throws IOException {
        String resourcePath = request.getParameter(PARAM_RESOURCE_PATH);
        String additionalPrompt = request.getParameter(PARAM_ADDITIONAL_PROMPT);
        List additionalUrls = new ArrayList(Stream.of(request.getParameterValues(PARAM_ADDITIONAL_URLS))
                .filter(s -> s != null && !s.trim().isEmpty())
                .flatMap(s -> Stream.of(s.split("\\s+")))
                .map(URI::create)
                .collect(Collectors.toList()));
        try (JsonWriter writer = gson.newJsonWriter(response.getWriter())) {
            try {
                String backgroundInformation = collectBackgroundInfoPaths(request, response, additionalUrls);
                if (request.getParameter(PARAM_BACKGROUND_INFORMATION) != null) {
                    backgroundInformation = backgroundInformation + request.getParameter(PARAM_BACKGROUND_INFORMATION);
                }
                Resource resource = request.getResourceResolver().getResource(resourcePath);
                if (resource == null) {
                    throw new IllegalArgumentException("Resource at " + resourcePath + " does not exist.");
                }
                boolean changed = aiTemplatingService.replacePromptsInResource(resource, additionalPrompt, additionalUrls, backgroundInformation);
                if (changed) request.getResourceResolver().commit();
                writeToResponse(writer, response, true, changed, (changed ? "Successfully " : "No changes made: ") +
                        "replacing prompts in resource " + resourcePath);
            } catch (Exception e) {
                LOG.error("Error replacing prompts in resource " + resourcePath, e);
                writeToResponse(writer, response, false, false, "Error replacing prompts in resource " + resourcePath + ": " + e);
            }
        }
    }

    /**
     * If some of the URLs are paths, we read the markdown for those pages. This cannot be done in the service easily since
     * we need the request and response objects for generating the markdown.
     */
    protected String collectBackgroundInfoPaths(SlingHttpServletRequest request, SlingHttpServletResponse response, List additionalUrls) {
        StringBuilder backgroundInformation = new StringBuilder();
        Iterator additionalUrlsIterator = additionalUrls.iterator();
        while (additionalUrlsIterator.hasNext()) {
            URI uri = additionalUrlsIterator.next();
            if (uri.toString().startsWith("/")) { // assume it's a JCR path.
                String path = uri.toString();
                if (path.endsWith("/")) path = path.substring(0, path.length() - 1);
                if (path.endsWith(".html")) path = path.substring(0, path.length() - 5);
                Resource resource = request.getResourceResolver().getResource(path);
                if (resource != null) {
                    String markdown = approximateMarkdownService.approximateMarkdown(resource, request, response);
                    if (StringUtils.isNotBlank(markdown)) {
                        backgroundInformation.append(markdown).append("\n\n");
                    } else {
                        throw new IllegalArgumentException("Resource at " + path + " does not contain any text.");
                    }
                    additionalUrlsIterator.remove();
                }
            }
        }
        return backgroundInformation.toString();
    }

    protected void writeToResponse(JsonWriter writer, SlingHttpServletResponse response, boolean success,
                                   boolean changes, String message) throws IOException {
        response.setContentType("application/json");
        response.setCharacterEncoding("UTF-8");
        writer.beginObject();
        writer.name("success").value(success);
        writer.name("changes").value(changes);
        writer.name("message").value(message);
        writer.endObject();
        writer.flush();
    }

    protected void resetToPrompts(SlingHttpServletRequest request, SlingHttpServletResponse response) throws
            IOException {
        String resourcePath = request.getParameter(PARAM_RESOURCE_PATH);
        try (JsonWriter writer = gson.newJsonWriter(response.getWriter())) {
            try {
                Resource resource = request.getResourceResolver().getResource(resourcePath);
                boolean changed = aiTemplatingService.resetToPrompts(resource);
                if (changed) request.getResourceResolver().commit();
                writeToResponse(writer, response, true, changed, (changed ? "Successfully " : "No changes made: ") +
                        "reset resource " + resourcePath + " to prompts");
            } catch (Exception e) {
                LOG.error("Error resetting resource " + resourcePath + " to prompts", e);
                writeToResponse(writer, response, false, false, "Error resetting resource " + resourcePath + " to prompts: " + e);
            }
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy