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

io.streamnative.pulsar.handlers.kop.schemaregistry.resources.SubjectResource Maven / Gradle / Ivy

There is a newer version: 4.0.0.10
Show newest version
/**
 * 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.
 */
package io.streamnative.pulsar.handlers.kop.schemaregistry.resources;

import io.netty.handler.codec.http.FullHttpRequest;
import io.netty.handler.codec.http.HttpResponseStatus;
import io.streamnative.pulsar.handlers.kop.schemaregistry.HttpJsonRequestProcessor;
import io.streamnative.pulsar.handlers.kop.schemaregistry.SchemaRegistryHandler;
import io.streamnative.pulsar.handlers.kop.schemaregistry.SchemaRegistryRequestAuthenticator;
import io.streamnative.pulsar.handlers.kop.schemaregistry.exceptions.Errors;
import io.streamnative.pulsar.handlers.kop.schemaregistry.model.CompatibilityChecker;
import io.streamnative.pulsar.handlers.kop.schemaregistry.model.LookupFilter;
import io.streamnative.pulsar.handlers.kop.schemaregistry.model.Schema;
import io.streamnative.pulsar.handlers.kop.schemaregistry.model.SchemaStorage;
import io.streamnative.pulsar.handlers.kop.schemaregistry.model.SchemaStorageAccessor;
import io.streamnative.pulsar.handlers.kop.schemaregistry.model.impl.SchemaStorageException;
import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.CompletableFuture;
import java.util.concurrent.CompletionException;
import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.Getter;
import org.apache.commons.collections4.CollectionUtils;

public class SubjectResource extends AbstractResource {

    public SubjectResource(SchemaStorageAccessor schemaStorageAccessor,
                           SchemaRegistryRequestAuthenticator schemaRegistryRequestAuthenticator) {
        super(schemaStorageAccessor, schemaRegistryRequestAuthenticator);
    }

    @Override
    public void register(SchemaRegistryHandler schemaRegistryHandler) {
        schemaRegistryHandler.addProcessor(new CreateNewSchema());
        schemaRegistryHandler.addProcessor(new GetAllSubjects());
        schemaRegistryHandler.addProcessor(new GetLatestVersion());
        schemaRegistryHandler.addProcessor(new GetAllVersions());
        schemaRegistryHandler.addProcessor(new DeleteSubject());
        schemaRegistryHandler.addProcessor(new GetSchemaBySubjectAndVersion());
        schemaRegistryHandler.addProcessor(new GetRawSchemaBySubjectAndVersion());
        schemaRegistryHandler.addProcessor(new LookupSchema());
    }

    @AllArgsConstructor
    @Getter
    public static class GetSchemaBySubjectAndVersionResponse {
        private int id;
        private String schema;
        private String subject;
        private int version;
    }

    @Data
    public static final class CreateSchemaRequest {
        String schema;
        String schemaType = "AVRO";
        List references = new ArrayList<>();

        @Data
        public static final class SchemaReference {
            String name;
            String subject;
            String version;
        }
    }

    @Data
    @AllArgsConstructor
    public static final class CreateSchemaResponse {
        int id;
    }

    @Data
    @AllArgsConstructor
    public static final class LookupResponseForSubject {
        int id;
        int version;
        String subject;
        String schema;
    }

    // GET /subjects
    public class GetAllSubjects extends HttpJsonRequestProcessor> {

        public GetAllSubjects() {
            super(Void.class, "/subjects", GET);
        }

        @Override
        protected CompletableFuture> processRequest(Void payload, List patternGroups,
                                                                 FullHttpRequest request)
                throws Exception {
            SchemaStorage schemaStorage = getSchemaStorage(request);
            return schemaStorage.getAllSubjects();
        }

    }

    // GET /subjects/test/versions
    public class GetAllVersions extends HttpJsonRequestProcessor> {

        public GetAllVersions() {
            super(Void.class, "/subjects/" + STRING_PATTERN + "/versions", GET);
        }

        @Override
        protected CompletableFuture> processRequest(Void payload, List patternGroups,
                                                                  FullHttpRequest request)
                throws Exception {
            SchemaStorage schemaStorage = getSchemaStorage(request);
            String subject = getString(0, patternGroups);
            CompletableFuture> versions = schemaStorage.getAllVersionsForSubject(subject);
            return versions.thenApply(v -> {
                if (v.isEmpty()) {
                    throw Errors.subjectNotFoundException(subject);
                }
                return v;
            });
        }

    }

    // GET /subjects/test/versions/latest
    public class GetLatestVersion extends HttpJsonRequestProcessor {

        public GetLatestVersion() {
            super(Void.class, "/subjects/" + STRING_PATTERN + "/versions/latest", GET);
        }

        @Override
        protected CompletableFuture processRequest(Void payload,
                                                                                         List patternGroups,
                                                                                         FullHttpRequest request)
                throws Exception {
            SchemaStorage schemaStorage = getSchemaStorage(request);
            String subject = getString(0, patternGroups);
            CompletableFuture> versions = schemaStorage.getAllVersionsForSubject(subject);
            return versions.thenCompose(v -> {
                if (v.isEmpty()) {
                    return CompletableFuture.completedFuture(null);
                }
                return schemaStorage.findSchemaBySubjectAndVersion(subject, v.get(v.size() - 1))
                        .thenApply(s -> s == null ? null : new GetSchemaBySubjectAndVersionResponse(
                                s.getId(),
                                s.getSchemaDefinition(),
                                s.getSubject(), s.getVersion()));
            });
        }

    }

    // DELETE /subjects/(string: subject)
    public class DeleteSubject extends HttpJsonRequestProcessor> {

        public DeleteSubject() {
            super(Void.class, "/subjects/" + STRING_PATTERN, DELETE);
        }

        @Override
        protected CompletableFuture> processRequest(Void payload, List patternGroups,
                                                                  FullHttpRequest request)
                throws Exception {
            SchemaStorage schemaStorage = getSchemaStorage(request);
            String subject = getString(0, patternGroups);

            CompletableFuture> versions = schemaStorage.deleteSubject(subject);
            return versions.thenApply(v -> {
                if (v.isEmpty()) {
                    return null;
                }
                return v;
            });
        }

    }

    // GET /subjects/(string: subject)/versions/(versionId: version)
    public class GetSchemaBySubjectAndVersion
            extends HttpJsonRequestProcessor {

        public GetSchemaBySubjectAndVersion() {
            super(Void.class, "/subjects/" + STRING_PATTERN + "/versions/" + INT_PATTERN, GET);
        }

        @Override
        protected CompletableFuture processRequest(Void payload,
                                                                                         List patternGroups,
                                                                                         FullHttpRequest request)
                throws Exception {
            String subject = getString(0, patternGroups);
            int version = getInt(1, patternGroups);
            SchemaStorage schemaStorage = getSchemaStorage(request);
            CompletableFuture schema = schemaStorage.findSchemaBySubjectAndVersion(subject, version);
            return schema.thenApply(s -> {
                if (s == null) {
                    return null;
                }
                return new GetSchemaBySubjectAndVersionResponse(s.getId(), s.getSchemaDefinition(), s.getSubject(),
                        s.getVersion());
            });
        }

    }

    // GET /subjects/(string: subject)/versions/(versionId: version)
    public class GetRawSchemaBySubjectAndVersion extends HttpJsonRequestProcessor {

        public GetRawSchemaBySubjectAndVersion() {
            super(Void.class, "/subjects/" + STRING_PATTERN + "/versions/" + INT_PATTERN + "/schema", GET);
        }

        @Override
        protected CompletableFuture processRequest(Void payload, List patternGroups,
                                                           FullHttpRequest request)
                throws Exception {
            String subject = getString(0, patternGroups);
            int version = getInt(1, patternGroups);
            SchemaStorage schemaStorage = getSchemaStorage(request);
            CompletableFuture schema = schemaStorage.findSchemaBySubjectAndVersion(subject, version);
            return schema.thenApply(s -> {
                if (s == null) {
                    return null;
                }
                return s.getSchemaDefinition();
            });
        }

    }

    // POST /subjects/(string: subject)/versions
    public class CreateNewSchema extends HttpJsonRequestProcessor {

        public CreateNewSchema() {
            super(CreateSchemaRequest.class, "/subjects/" + STRING_PATTERN + "/versions", POST);
        }

        @Override
        protected CompletableFuture processRequest(CreateSchemaRequest payload,
                                                                         List patternGroups,
                                                                         FullHttpRequest request)
                throws Exception {
            String subject = getString(0, patternGroups);
            SchemaStorage schemaStorage = getSchemaStorage(request);
            CompletableFuture schema = schemaStorage.createSchemaVersion(subject,
                    payload.schemaType, payload.schema);
            return schema.thenApply(s -> new CreateSchemaResponse(s.getId())).exceptionally(err -> {
                while (err instanceof CompletionException) {
                    err = err.getCause();
                }
                if (err instanceof CompatibilityChecker.IncompatibleSchemaChangeException) {
                    throw new CompletionException(
                            new SchemaStorageException(err.getMessage(), HttpResponseStatus.CONFLICT));
                } else {
                    throw new CompletionException(err);
                }
            });
        }

    }

    // POST /subjects/(string: subject)
    public class LookupSchema
            extends HttpJsonRequestProcessor {

        public LookupSchema() {
            super(CreateSchemaRequest.class, "/subjects/" + STRING_PATTERN, POST);
        }

        @Override
        protected CompletableFuture processRequest(CreateSchemaRequest payload,
                                                                             List patternGroups,
                                                                             FullHttpRequest request)
                throws Exception {
            final String subject = getString(0, patternGroups);
            return getSchemaStorage(request).getSchemasForSubject(subject, LookupFilter.DEFAULT).thenApply(list -> {
                if (CollectionUtils.isEmpty(list)) {
                    throw Errors.subjectNotFoundException(subject);
                }
                for (Schema schema : list) {
                    if (schema.getSchemaDefinition().equals(payload.getSchema())) {
                        return new LookupResponseForSubject(schema.getId(), schema.getVersion(), schema.getSubject(),
                                schema.getSchemaDefinition());
                    }
                }
                throw Errors.schemaNotFoundException();
            }).exceptionally(err -> {
                while (err instanceof CompletionException) {
                    err = err.getCause();
                }
                if (err instanceof CompatibilityChecker.IncompatibleSchemaChangeException) {
                    throw new CompletionException(
                            new SchemaStorageException(err.getMessage(), HttpResponseStatus.CONFLICT));
                } else {
                    throw new CompletionException(err);
                }
            });
        }

    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy