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

io.thestencil.quarkus.ide.services.HandlerComposer Maven / Gradle / Ivy

There is a newer version: 1.148.40
Show newest version
package io.thestencil.quarkus.ide.services;

/*-
 * #%L
 * quarkus-stencil-ide-services
 * %%
 * Copyright (C) 2021 Copyright 2021 ReSys OÜ
 * %%
 * Licensed under the Apache License, Version 2.0 (the "License");
 * 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 "AS IS" 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%
 */

import java.io.IOException;
import java.nio.charset.StandardCharsets;
import java.util.List;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import com.fasterxml.jackson.core.type.TypeReference;
import com.fasterxml.jackson.databind.ObjectMapper;

import io.quarkus.security.identity.CurrentIdentityAssociation;
import io.quarkus.vertx.http.runtime.CurrentVertxRequest;
import io.smallrye.mutiny.Uni;
import io.thestencil.client.api.ImmutableArticleMutator;
import io.thestencil.client.api.ImmutableCreateArticle;
import io.thestencil.client.api.ImmutableCreateLink;
import io.thestencil.client.api.ImmutableCreateLocale;
import io.thestencil.client.api.ImmutableCreatePage;
import io.thestencil.client.api.ImmutableCreateRelease;
import io.thestencil.client.api.ImmutableCreateTemplate;
import io.thestencil.client.api.ImmutableCreateWorkflow;
import io.thestencil.client.api.ImmutableLinkArticlePage;
import io.thestencil.client.api.ImmutableLinkMutator;
import io.thestencil.client.api.ImmutableLocaleMutator;
import io.thestencil.client.api.ImmutableSiteState;
import io.thestencil.client.api.ImmutableTemplateMutator;
import io.thestencil.client.api.ImmutableWorkflowArticlePage;
import io.thestencil.client.api.ImmutableWorkflowMutator;
import io.thestencil.client.api.MigrationBuilder.Sites;
import io.thestencil.client.api.StencilComposer.SiteState;
import io.thestencil.client.api.StencilClient.VersionInfo;
import io.thestencil.client.api.UpdateBuilder.PageMutator;
import io.thestencil.client.spi.beans.SitesBean;
import io.thestencil.client.spi.composer.HandlerStatusCodes;
import io.vertx.core.buffer.Buffer;
import io.vertx.core.http.HttpHeaders;
import io.vertx.core.http.HttpMethod;
import io.vertx.core.http.HttpServerResponse;
import io.vertx.core.json.JsonObject;
import io.vertx.ext.web.RoutingContext;


public class HandlerComposer extends HandlerTemplate {
  private static final Logger LOGGER = LoggerFactory.getLogger(HandlerComposer.class);
  public HandlerComposer(CurrentIdentityAssociation currentIdentityAssociation, CurrentVertxRequest currentVertxRequest) {
    super(currentIdentityAssociation, currentVertxRequest);
  }

  @Override
  protected void handleResource(RoutingContext event, HttpServerResponse response, HandlerContext ctx, ObjectMapper objectMapper) throws IOException {
    response.headers().set(HttpHeaders.CONTENT_TYPE, "application/json; charset=UTF-8");
    final var path = getPath(event);
    
    if(path.endsWith(ctx.getPaths().getServicePath())) {
    	doSite(event, response, ctx, objectMapper);
    } else if (path.startsWith(ctx.getPaths().getMigrationPath())) {
      doMigration(event, response, ctx, objectMapper);
    } else if (path.startsWith(ctx.getPaths().getArticlesPath())) {
      doArticles(event, response, ctx, objectMapper);
    } else if(path.startsWith(ctx.getPaths().getLinksPath())){
      doLinks(event, response, ctx, objectMapper);
    } else if(path.startsWith(ctx.getPaths().getLocalePath())) {
      doLocales(event, response, ctx, objectMapper);
    } else if(path.startsWith(ctx.getPaths().getReleasesPath())) {
      doReleases(event, response, ctx, objectMapper);
    } else if(path.startsWith(ctx.getPaths().getTemplatesPath())) {
      doTemplates(event, response, ctx, objectMapper);
    } else if(path.startsWith(ctx.getPaths().getWorkflowsPath())) {
      doWorkflows(event, response, ctx, objectMapper);
    } else if(path.startsWith(ctx.getPaths().getPagesPath())) {
      doPages(event, response, ctx, objectMapper);
    } else if(path.startsWith(ctx.getPaths().getVersionPath())) {
      doVersion(event, response, ctx, objectMapper);
    } else {
      HandlerStatusCodes.catch404("unsupported action", response);
    }
  }
  
  public String getPath(RoutingContext event) {
    final var path = event.normalizedPath();
    
    return path.endsWith("/") ? path.substring(0, path.length() -1) : path;
  }
  
  public void doSite(RoutingContext event, HttpServerResponse response, HandlerContext ctx, ObjectMapper objectMapper) {
	  final var client = ctx.getClient();
    if (event.request().method() == HttpMethod.POST) {
  		client.create().repo()
  		.onItem().transform(data -> JsonObject.mapFrom(data).toBuffer())
  		.onFailure().invoke(e -> HandlerStatusCodes.catch422(e, response))
  		.subscribe().with(data -> response.end(data));
  
    } else if(event.request().method() == HttpMethod.GET) { 
  		subscribe(
  		    client.query().head(), 
  		    response, ctx, objectMapper);
    } else {
      	HandlerStatusCodes.catch404("unsupported repository action", response);
  	}
  }
  
  public void doArticles(RoutingContext event, HttpServerResponse response, HandlerContext ctx, ObjectMapper objectMapper) {
    final var client = ctx.getClient();    
    if (event.request().method() == HttpMethod.POST) {
      subscribe(
          client.create().article(read(event, objectMapper, ImmutableCreateArticle.class)),
          response, ctx, objectMapper);
      
    } else if(event.request().method() == HttpMethod.PUT) {
      subscribe(
          client.update().article(read(event, objectMapper, ImmutableArticleMutator.class)),
          response, ctx, objectMapper);
      
    } else if(event.request().method() == HttpMethod.DELETE) {
      subscribe(
          client.delete().article(event.pathParam("id")),
          response, ctx, objectMapper);
    } else {
      HandlerStatusCodes.catch404("unsupported article action", response);
    }
  }
  
  public void doLinks(RoutingContext event, HttpServerResponse response, HandlerContext ctx, ObjectMapper objectMapper) {
    final var client = ctx.getClient();
    if (event.request().method() == HttpMethod.POST) {
      subscribe(
          client.create().link(read(event, objectMapper, ImmutableCreateLink.class)),
          response, ctx, objectMapper);
      
    } else if(event.request().method() == HttpMethod.PUT) {
      subscribe(
          client.update().link(read(event, objectMapper, ImmutableLinkMutator.class)),
          response, ctx, objectMapper);
    } else if(event.request().method() == HttpMethod.DELETE) {
      
      final var linkId = event.pathParam("id");
      final var articleId = event.queryParam("articleId");
      
      if(articleId.isEmpty()) {
        subscribe(
            client.delete().link(linkId),
            response, ctx, objectMapper);
      } else {
        subscribe(
            client.delete().linkArticlePage(ImmutableLinkArticlePage.builder()
                .articleId(articleId.iterator().next())
                .linkId(linkId)
                .build()),
            response, ctx, objectMapper);  
      }
    } else {
      HandlerStatusCodes.catch404("unsupported links action", response);
    }
  }
  
  public void doTemplates(RoutingContext event, HttpServerResponse response, HandlerContext ctx, ObjectMapper objectMapper) {
    final var client = ctx.getClient();    
    if (event.request().method() == HttpMethod.POST) {
      subscribe(
          client.create().template(read(event, objectMapper, ImmutableCreateTemplate.class)),
          response, ctx, objectMapper);
      
    } else if(event.request().method() == HttpMethod.PUT) {
      subscribe(
          client.update().template(read(event, objectMapper, ImmutableTemplateMutator.class)),
          response, ctx, objectMapper);
    } else if(event.request().method() == HttpMethod.DELETE) {
      subscribe(
          client.delete().template(event.pathParam("id")),
          response, ctx, objectMapper);
    } else {
      HandlerStatusCodes.catch404("unsupported template action", response);
    }
  }
  
  public void doReleases(RoutingContext event, HttpServerResponse response, HandlerContext ctx, ObjectMapper objectMapper) {
    final var client = ctx.getClient();

    if (event.request().method() == HttpMethod.POST) {
      subscribe(
          client.create().release(read(event, objectMapper, ImmutableCreateRelease.class)), 
          response, ctx, objectMapper);
    } else if(event.request().method() == HttpMethod.GET) {
      subscribe(
          client.query().release(event.pathParam("id")), 
          response, ctx, objectMapper);        
    } else if(event.request().method() == HttpMethod.DELETE) {
      subscribe(
          client.delete().release(event.pathParam("id")),
          response, ctx, objectMapper);
    } else {
      HandlerStatusCodes.catch404("unsupported release action", response);
    }
  }
  
  public void doLocales(RoutingContext event, HttpServerResponse response, HandlerContext ctx, ObjectMapper objectMapper) {
    final var client = ctx.getClient();
    if (event.request().method() == HttpMethod.POST) {
      subscribe(
          client.create().locale(read(event, objectMapper, ImmutableCreateLocale.class)), 
          response, ctx, objectMapper);
      
    } else if(event.request().method() == HttpMethod.PUT) {
      subscribe(
          client.update().locale(read(event, objectMapper, ImmutableLocaleMutator.class)),
          response, ctx, objectMapper);
    } else if(event.request().method() == HttpMethod.DELETE) {
      subscribe(
          client.delete().locale(event.pathParam("id")),
          response, ctx, objectMapper);
    } else {
      HandlerStatusCodes.catch404("unsupported locale action", response);
    }
  }
  
  public void doWorkflows(RoutingContext event, HttpServerResponse response, HandlerContext ctx, ObjectMapper objectMapper) {
    final var client = ctx.getClient();
    if (event.request().method() == HttpMethod.POST) {
      subscribe(
          client.create().workflow(read(event, objectMapper, ImmutableCreateWorkflow.class)), 
          response, ctx, objectMapper);
      
    } else if(event.request().method() == HttpMethod.PUT) {
      subscribe(
          client.update().workflow(read(event, objectMapper, ImmutableWorkflowMutator.class)),
          response, ctx, objectMapper);
    } else if(event.request().method() == HttpMethod.DELETE) {
            
      final var workflowId = event.pathParam("id");
      final var articleId = event.queryParam("articleId");
      
      if(articleId.isEmpty()) {
        subscribe(
            client.delete().workflow(workflowId),
            response, ctx, objectMapper);
        
      } else {
        subscribe(
            client.delete().workflowArticlePage(ImmutableWorkflowArticlePage.builder()
                .articleId(articleId.iterator().next())
                .workflowId(workflowId)
                .build()),
            response, ctx, objectMapper);  
      }

    } else {
      HandlerStatusCodes.catch404("unsupported workflow action", response);
    }
  }
  
  public void doPages(RoutingContext event, HttpServerResponse response, HandlerContext ctx, ObjectMapper objectMapper) {
    final var client = ctx.getClient();
    
    if (event.request().method() == HttpMethod.POST) {
      subscribe(
          client.create().page(read(event, objectMapper, ImmutableCreatePage.class)),
          response, ctx, objectMapper);
    } else if(event.request().method() == HttpMethod.PUT) {
      
      try {
        List pages = objectMapper.readValue(event.getBody().getBytes(), new TypeReference>(){});
        
        subscribe(
            client.update().pages(pages),
            response, ctx, objectMapper);
      } catch(IOException e) {
        throw new RuntimeException(e.getMessage(), e);
      }
      
    } else if(event.request().method() == HttpMethod.DELETE) {
      subscribe(
          client.delete().page(event.pathParam("id")),
          response, ctx, objectMapper);

    } else {
      HandlerStatusCodes.catch404("unsupported page action", response);
    }
  }

  public void doMigration(RoutingContext event, HttpServerResponse response, HandlerContext ctx, ObjectMapper objectMapper) {
    final var client = ctx.getClient();
    if (event.request().method() == HttpMethod.POST) {
      byte[] body = event.getBody().getBytes();
      
      
      final var sites = parseSites(ctx, body, objectMapper);
      if(sites != null) {
        subscribe(
            client.migration().importData(sites), 
            response, ctx, objectMapper);
        return;
      }
      
      final var release = parseSiteState(body, objectMapper);
      if(release != null) {
        subscribe(
            client.migration().importData(release), 
            response, ctx, objectMapper);
        return;
      }
      HandlerStatusCodes.catch422("No static content or release to parse", response);
    } else {
      HandlerStatusCodes.catch404("unsupported migration action", response);
    }
  }

  public void doVersion(RoutingContext event, HttpServerResponse response, HandlerContext ctx, ObjectMapper objectMapper) throws IOException {
    final var client = ctx.getClient();
    if (event.request().method() == HttpMethod.GET) {
      VersionInfo version = client.version().version();
      response.end(objectMapper.writeValueAsString(version));
    } else {
      HandlerStatusCodes.catch404("unsupported version action", response);
    }
  }
  
  private SiteState parseSiteState(byte[] body, ObjectMapper objectMapper) {
    try {
      return objectMapper.readValue(body, ImmutableSiteState.class);
    } catch(IOException ex1) {
      LOGGER.error("failed to parse site for migration, " + ex1.getMessage(), ex1);
    }
    return null;
  }
  
  private Sites parseSites(HandlerContext ctx, byte[] body, ObjectMapper objectMapper) {
    final var client = ctx.getClient();    
    Sites site = null;
    try {
      site = objectMapper.readValue(body, SitesBean.class);
      
      if(site == null || site.getSites() == null  || site.getSites().isEmpty()) {
        final var md = client
            .markdown().json(new String(body, StandardCharsets.UTF_8), true)
            .build();

        site = client
            .sites().imagePath("/images").created(1l)
            .source(md)
            .build();
      }

      return site;
    } catch(IOException ex1) {
      LOGGER.error("failed to parse site for migration, " + ex1.getMessage(), ex1);
    }
    return null;
  }
  
  public  T read(RoutingContext event, ObjectMapper objectMapper, Class type) {
    
   // return new JsonObject(event.getBody()).mapTo(type);
    try {
      return objectMapper.readValue(event.getBody().getBytes(), type);
    } catch(IOException e) {
      throw new RuntimeException(e.getMessage(), e);
    }
  }
  
  public  List readList(RoutingContext event, ObjectMapper objectMapper, Class type) {
    
   // return new JsonObject(event.getBody()).mapTo(type);
    try {
      return objectMapper.readValue(event.getBody().getBytes(), new TypeReference>(){});
    } catch(IOException e) {
      throw new RuntimeException(e.getMessage(), e);
    }
  }

  public  void subscribe(Uni uni, HttpServerResponse response, HandlerContext ctx, ObjectMapper objectMapper) {
    uni.onItem().transform(data -> {
      try {
        return Buffer.buffer(objectMapper.writeValueAsBytes(data));
      } catch(IOException e) {
        throw new RuntimeException(e.getMessage(), e);
      }
    })
    .onFailure().invoke(e -> HandlerStatusCodes.catch422(e, response))
    .subscribe().with(data -> response.end(data)); 
  }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy