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

org.apache.solr.filestore.PackageStoreAPI Maven / Gradle / Ivy

There is a newer version: 9.7.0
Show newest version
/*
 * Licensed to the Apache Software Foundation (ASF) under one or more
 * contributor license agreements.  See the NOTICE file distributed with
 * this work for additional information regarding copyright ownership.
 * The ASF licenses this file to You 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 org.apache.solr.filestore;

import java.io.IOException;
import java.io.InputStream;
import java.lang.invoke.MethodHandles;
import java.nio.ByteBuffer;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.function.Consumer;
import java.util.function.Supplier;

import org.apache.commons.codec.digest.DigestUtils;
import org.apache.solr.api.EndPoint;
import org.apache.solr.client.solrj.SolrRequest;
import org.apache.solr.common.MapWriter;
import org.apache.solr.common.SolrException;
import org.apache.solr.common.params.CommonParams;
import org.apache.solr.common.params.ModifiableSolrParams;
import org.apache.solr.common.params.SolrParams;
import org.apache.solr.common.util.ContentStream;
import org.apache.solr.common.util.StrUtils;
import org.apache.solr.common.util.Utils;
import org.apache.solr.core.BlobRepository;
import org.apache.solr.core.CoreContainer;
import org.apache.solr.core.SolrCore;
import org.apache.solr.pkg.PackageAPI;
import org.apache.solr.request.SolrQueryRequest;
import org.apache.solr.response.SolrQueryResponse;
import org.apache.solr.security.PermissionNameProvider;
import org.apache.solr.util.CryptoKeys;
import org.apache.solr.util.SimplePostTool;
import org.apache.zookeeper.CreateMode;
import org.apache.zookeeper.KeeperException;
import org.apache.zookeeper.server.ByteBufferInputStream;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import static java.nio.charset.StandardCharsets.UTF_8;
import static org.apache.solr.handler.ReplicationHandler.FILE_STREAM;


public class PackageStoreAPI {
  private static final Logger log = LoggerFactory.getLogger(MethodHandles.lookup().lookupClass());
  public static final String PACKAGESTORE_DIRECTORY = "filestore";
  public static final String TRUSTED_DIR = "_trusted_";
  public static final String KEYS_DIR = "/_trusted_/keys";


  private final CoreContainer coreContainer;
  PackageStore packageStore;
  public final FSRead readAPI = new FSRead();
  public final FSWrite writeAPI = new FSWrite();

  public PackageStoreAPI(CoreContainer coreContainer) {
    this.coreContainer = coreContainer;
    packageStore = new DistribPackageStore(coreContainer);
  }

  public PackageStore getPackageStore() {
    return packageStore;
  }

  /**
   * get a list of nodes randomly shuffled
   * * @lucene.internal
   */
  public ArrayList shuffledNodes() {
    Set liveNodes = coreContainer.getZkController().getZkStateReader().getClusterState().getLiveNodes();
    ArrayList l = new ArrayList<>(liveNodes);
    l.remove(coreContainer.getZkController().getNodeName());
    Collections.shuffle(l, BlobRepository.RANDOM);
    return l;
  }

  public void validateFiles(List files, boolean validateSignatures, Consumer errs) {
    for (String path : files) {
      try {
        PackageStore.FileType type = packageStore.getType(path, true);
        if (type != PackageStore.FileType.FILE) {
          errs.accept("No such file: " + path);
          continue;
        }

        packageStore.get(path, entry -> {
          if (entry.getMetaData().signatures == null ||
              entry.getMetaData().signatures.isEmpty()) {
            errs.accept(path + " has no signature");
            return;
          }
          if (validateSignatures) {
            try {
              packageStore.refresh(KEYS_DIR);
              validate(entry.meta.signatures, entry, false);
            } catch (Exception e) {
              log.error("Error validating package artifact", e);
              errs.accept(e.getMessage());
            }
          }
        }, false);
      } catch (Exception e) {
        log.error("Error reading file ", e);
        errs.accept("Error reading file " + path + " " + e.getMessage());
      }
    }

  }

  public class FSWrite {

    static final String TMP_ZK_NODE = "/packageStoreWriteInProgress";

    @EndPoint(
        path = "/cluster/files/*",
        method = SolrRequest.METHOD.PUT,
        permission = PermissionNameProvider.Name.FILESTORE_WRITE_PERM)
    public void upload(SolrQueryRequest req, SolrQueryResponse rsp) {
      if (!coreContainer.getPackageLoader().getPackageAPI().isEnabled()) {
        throw new RuntimeException(PackageAPI.ERR_MSG);
      }
      try {
        coreContainer.getZkController().getZkClient().create(TMP_ZK_NODE, "true".getBytes(UTF_8),
            CreateMode.EPHEMERAL, true);

        Iterable streams = req.getContentStreams();
        if (streams == null) throw new SolrException(SolrException.ErrorCode.BAD_REQUEST, "no payload");
        String path = req.getPathTemplateValues().get("*");
        if (path == null) {
          throw new SolrException(SolrException.ErrorCode.BAD_REQUEST, "No path");
        }
        validateName(path, true);
        ContentStream stream = streams.iterator().next();
        try {
          ByteBuffer buf = SimplePostTool.inputStreamToByteArray(stream.getStream());
          List signatures = readSignatures(req, buf);
          MetaData meta = _createJsonMetaData(buf, signatures);
          PackageStore.FileType type = packageStore.getType(path, true);
          if(type != PackageStore.FileType.NOFILE) {
            throw new SolrException(SolrException.ErrorCode.BAD_REQUEST,  "Path already exists "+ path);
          }
          packageStore.put(new PackageStore.FileEntry(buf, meta, path));
          rsp.add(CommonParams.FILE, path);
        } catch (IOException e) {
          throw new SolrException(SolrException.ErrorCode.BAD_REQUEST, e);
        }
      } catch (InterruptedException e) {
        log.error("Unexpected error", e);
      } catch (KeeperException.NodeExistsException e) {
        throw new SolrException(SolrException.ErrorCode.SERVER_ERROR, "A write is already in process , try later");
      } catch (KeeperException e) {
        log.error("Unexpected error", e);
      } finally {
        try {
          coreContainer.getZkController().getZkClient().delete(TMP_ZK_NODE, -1, true);
        } catch (Exception e) {
          log.error("Unexpected error  ", e);
        }
      }
    }

    private List readSignatures(SolrQueryRequest req, ByteBuffer buf)
        throws SolrException, IOException {
      String[] signatures = req.getParams().getParams("sig");
      if (signatures == null || signatures.length == 0) return null;
      List sigs = Arrays.asList(signatures);
      packageStore.refresh(KEYS_DIR);
      validate(sigs, buf);
      return sigs;
    }

    private void validate(List sigs,
                          ByteBuffer buf) throws SolrException, IOException {
      Map keys = packageStore.getKeys();
      if (keys == null || keys.isEmpty()) {
        throw new SolrException(SolrException.ErrorCode.BAD_REQUEST,
            "package store does not have any keys");
      }
      CryptoKeys cryptoKeys = null;
      try {
        cryptoKeys = new CryptoKeys(keys);
      } catch (Exception e) {
        throw new SolrException(SolrException.ErrorCode.SERVER_ERROR,
            "Error parsing public keys in Package store");
      }
      for (String sig : sigs) {
        if (cryptoKeys.verify(sig, buf) == null) {
          throw new SolrException(SolrException.ErrorCode.BAD_REQUEST, "Signature does not match any public key : " + sig +" len: "+buf.limit()+  " content sha512: "+
              DigestUtils.sha512Hex(new ByteBufferInputStream(buf)));
        }

      }
    }

  }

  /**
   * Creates a JSON string with the metadata
   * @lucene.internal
   */
  public static MetaData _createJsonMetaData(ByteBuffer buf, List signatures) throws IOException {
    String sha512 = DigestUtils.sha512Hex(new ByteBufferInputStream(buf));
    Map vals = new HashMap<>();
    vals.put(MetaData.SHA512, sha512);
    if (signatures != null) {
      vals.put("sig", signatures);
    }
    return new MetaData(vals);
  }

  public class FSRead {
    @EndPoint(
        path = "/node/files/*",
        method = SolrRequest.METHOD.GET,
        permission = PermissionNameProvider.Name.FILESTORE_READ_PERM)
    public void read(SolrQueryRequest req, SolrQueryResponse rsp) {
      String path = req.getPathTemplateValues().get("*");
      String pathCopy = path;
      if (req.getParams().getBool("sync", false)) {
        try {
          packageStore.syncToAllNodes(path);
          return;
        } catch (IOException e) {
          throw new SolrException(SolrException.ErrorCode.BAD_REQUEST, "Error getting file ", e);
        }
      }
      String getFrom = req.getParams().get("getFrom");
      if (getFrom != null) {
        coreContainer.getUpdateShardHandler().getUpdateExecutor().submit(() -> {
          log.debug("Downloading file {}", pathCopy);
          try {
            packageStore.fetch(pathCopy, getFrom);
          } catch (Exception e) {
            log.error("Failed to download file: {}", pathCopy, e);
          }
          log.info("downloaded file: {}", pathCopy);
        });
        return;

      }
      if (path == null) {
        path = "";
      }

      PackageStore.FileType typ = packageStore.getType(path, false);
      if (typ == PackageStore.FileType.NOFILE) {
        rsp.add("files", Collections.singletonMap(path, null));
        return;
      }
      if (typ == PackageStore.FileType.DIRECTORY) {
        rsp.add("files", Collections.singletonMap(path, packageStore.list(path, null)));
        return;
      }
      if (req.getParams().getBool("meta", false)) {
        if (typ == PackageStore.FileType.FILE) {
          int idx = path.lastIndexOf('/');
          String fileName = path.substring(idx + 1);
          String parentPath = path.substring(0, path.lastIndexOf('/'));
          @SuppressWarnings({"rawtypes"})
          List l = packageStore.list(parentPath, s -> s.equals(fileName));
          rsp.add("files", Collections.singletonMap(path, l.isEmpty() ? null : l.get(0)));
          return;
        }
      } else {
        writeRawFile(req, rsp, path);
      }
    }

    private void writeRawFile(SolrQueryRequest req, SolrQueryResponse rsp, String path) {
      ModifiableSolrParams solrParams = new ModifiableSolrParams();
      solrParams.add(CommonParams.WT, FILE_STREAM);
      req.setParams(SolrParams.wrapDefaults(solrParams, req.getParams()));
      rsp.add(FILE_STREAM, (SolrCore.RawWriter) os -> {
        packageStore.get(path, (it) -> {
          try {
            org.apache.commons.io.IOUtils.copy(it.getInputStream(), os);
          } catch (IOException e) {
            throw new SolrException(SolrException.ErrorCode.SERVER_ERROR, "Error reading file" + path);
          }
        }, false);

      });
    }

  }

  public static class MetaData implements MapWriter {
    public static final String SHA512 = "sha512";
    String sha512;
    List signatures;
    Map otherAttribs;

    @SuppressWarnings({"unchecked"})
    public MetaData(@SuppressWarnings({"rawtypes"})Map m) {
      m = Utils.getDeepCopy(m, 3);
      this.sha512 = (String) m.remove(SHA512);
      this.signatures = (List) m.remove("sig");
      this.otherAttribs = m;
    }

    @Override
    public void writeMap(EntryWriter ew) throws IOException {
      ew.putIfNotNull("sha512", sha512);
      ew.putIfNotNull("sig", signatures);
      if (!otherAttribs.isEmpty()) {
        otherAttribs.forEach(ew.getBiConsumer());
      }
    }
  }

  static final String INVALIDCHARS = " /\\#&*\n\t%@~`=+^$> parts = StrUtils.splitSmart(path, '/', true);
    for (String part : parts) {
      if (part.charAt(0) == '.') {
        throw new SolrException(SolrException.ErrorCode.BAD_REQUEST, "cannot start with period");
      }
      for (int i = 0; i < part.length(); i++) {
        for (int j = 0; j < INVALIDCHARS.length(); j++) {
          if (part.charAt(i) == INVALIDCHARS.charAt(j))
            throw new SolrException(SolrException.ErrorCode.BAD_REQUEST, "Unsupported char in file name: " + part);
        }
      }
    }
    if (failForTrusted &&  TRUSTED_DIR.equals(parts.get(0))) {
      throw new SolrException(SolrException.ErrorCode.BAD_REQUEST, "trying to write into /_trusted_/ directory");
    }
  }

  /**Validate a file for signature
   *
   * @param sigs the signatures. atleast one should succeed
   * @param entry The file details
   * @param isFirstAttempt If there is a failure
   */
  public void validate(List sigs,
                       PackageStore.FileEntry entry,
                       boolean isFirstAttempt) throws SolrException, IOException {
    if (!isFirstAttempt) {
      //we are retrying because last validation failed.
      // get all keys again and try again
      packageStore.refresh(KEYS_DIR);
    }

    Map keys = packageStore.getKeys();
    if (keys == null || keys.isEmpty()) {
      if(isFirstAttempt) {
        validate(sigs, entry, false);
        return;
      }
      throw new SolrException(SolrException.ErrorCode.BAD_REQUEST,
          "Packagestore does not have any public keys");
    }
    CryptoKeys cryptoKeys = null;
    try {
      cryptoKeys = new CryptoKeys(keys);
    } catch (Exception e) {
      throw new SolrException(SolrException.ErrorCode.SERVER_ERROR,
          "Error parsing public keys in ZooKeeper");
    }
    for (String sig : sigs) {
      Supplier errMsg = () -> "Signature does not match any public key : " + sig + "sha256 " + entry.getMetaData().sha512;
      if (entry.getBuffer() != null) {
        if (cryptoKeys.verify(sig, entry.getBuffer()) == null) {
          if(isFirstAttempt) {
            validate(sigs, entry, false);
            return;
          }
          throw new SolrException(SolrException.ErrorCode.BAD_REQUEST, errMsg.get());
        }
      } else {
        InputStream inputStream = entry.getInputStream();
        if (cryptoKeys.verify(sig, inputStream) == null) {
          if(isFirstAttempt)  {
            validate(sigs, entry, false);
            return;
          }
          throw new SolrException(SolrException.ErrorCode.BAD_REQUEST, errMsg.get());
        }

      }

    }
  }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy