com.github.horrorho.inflatabledonkey.cloud.clients.KeyBagClient.java Source code

Java tutorial

Introduction

Here is the source code for com.github.horrorho.inflatabledonkey.cloud.clients.KeyBagClient.java

Source

/*
 * The MIT License
 *
 * Copyright 2016 Ahseya.
 *
 * Permission is hereby granted, free of charge, to any person obtaining a copy
 * of this software and associated documentation files (the "Software"), to deal
 * in the Software without restriction, including without limitation the rights
 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
 * copies of the Software, and to permit persons to whom the Software is
 * furnished to do so, subject to the following conditions:
 *
 * The above copyright notice and this permission notice shall be included in
 * all copies or substantial portions of the Software.
 *
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
 * THE SOFTWARE.
 */
package com.github.horrorho.inflatabledonkey.cloud.clients;

import com.github.horrorho.inflatabledonkey.cloudkitty.CloudKitty;
import com.github.horrorho.inflatabledonkey.keybag.KeyBag;
import com.github.horrorho.inflatabledonkey.keybag.KeyBagFactory;
import com.github.horrorho.inflatabledonkey.pcs.xzone.ProtectionZone;
import com.github.horrorho.inflatabledonkey.protocol.CloudKit;
import com.google.protobuf.ByteString;
import java.io.IOException;
import java.util.List;
import java.util.Optional;
import java.util.function.BiFunction;
import net.jcip.annotations.Immutable;
import org.apache.http.client.HttpClient;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

/**
 * KeyBagClient.
 *
 * @author Ahseya
 */
@Immutable
public final class KeyBagClient {

    private static final Logger logger = LoggerFactory.getLogger(KeyBagClient.class);

    public static Optional<KeyBag> keyBag(HttpClient httpClient, CloudKitty kitty, ProtectionZone zone,
            String keyBagUUID) throws IOException {
        logger.debug("-- keyBag() - keybag UUID: {}", keyBagUUID);

        List<CloudKit.RecordRetrieveResponse> responses = kitty.recordRetrieveRequest(httpClient, "mbksync",
                "K:" + keyBagUUID);

        if (responses.size() != 1) {
            logger.warn("-- keyBag() - bad response list size: {}", responses);
            return Optional.empty();
        }

        CloudKit.ProtectionInfo protectionInfo = responses.get(0).getRecord().getProtectionInfo();

        Optional<ProtectionZone> optionalNewZone = zone.newProtectionZone(protectionInfo);
        if (!optionalNewZone.isPresent()) {
            logger.warn("-- keyBag() - failed to retrieve protection info");
            return Optional.empty();
        }
        ProtectionZone newZone = optionalNewZone.get();

        return keyBag(responses.get(0), newZone);
    }

    static Optional<KeyBag> keyBag(CloudKit.RecordRetrieveResponse response, ProtectionZone zone) {
        Optional<byte[]> keyBagData = field(response.getRecord(), "keybagData", zone::decrypt);
        if (!keyBagData.isPresent()) {
            logger.warn("-- keyBag() - failed to acquire key bag");
            return Optional.empty();
        }

        Optional<byte[]> secret = field(response.getRecord(), "secret", zone::decrypt);
        if (!secret.isPresent()) {
            logger.warn("-- keyBag() - failed to acquire key bag pass code");
            return Optional.empty();
        }

        KeyBag keyBag = KeyBagFactory.create(keyBagData.get(), secret.get());
        return Optional.of(keyBag);
    }

    static Optional<byte[]> field(CloudKit.Record record, String label,
            BiFunction<byte[], String, Optional<byte[]>> decrypt) {

        return record.getRecordFieldList().stream().filter(value -> value.getIdentifier().getName().equals(label))
                .map(CloudKit.RecordField::getValue).map(CloudKit.RecordFieldValue::getBytesValue)
                .map(ByteString::toByteArray).map(bs -> decrypt.apply(bs, label)).filter(Optional::isPresent)
                .map(Optional::get).findFirst();
    }
}