package org.nick.androidkeystore.android.security; /* * Copyright (C) 2009 The Android Open Source Project * * 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. */ import java.io.IOException; import java.io.InputStream; import java.io.OutputStream; import java.io.UnsupportedEncodingException; import java.util.ArrayList; import android.net.LocalSocket; import android.net.LocalSocketAddress; /** * @hide This should not be made public in its present form because it * assumes that private and secret key bytes are available and would * preclude the use of hardware crypto. */ public class KeyStore { // ResponseCodes public static final int NO_ERROR = 1; public static final int LOCKED = 2; public static final int UNINITIALIZED = 3; public static final int SYSTEM_ERROR = 4; public static final int PROTOCOL_ERROR = 5; public static final int PERMISSION_DENIED = 6; public static final int KEY_NOT_FOUND = 7; public static final int VALUE_CORRUPTED = 8; public static final int UNDEFINED_ACTION = 9; public static final int WRONG_PASSWORD = 10; // States public enum State { UNLOCKED, LOCKED, UNINITIALIZED }; private static final LocalSocketAddress sAddress = new LocalSocketAddress( "keystore", LocalSocketAddress.Namespace.RESERVED); private int mError = NO_ERROR; KeyStore() { } public static KeyStore getInstance() { return new KeyStore(); } public State state() { execute('t'); switch (mError) { case NO_ERROR: return State.UNLOCKED; case LOCKED: return State.LOCKED; case UNINITIALIZED: return State.UNINITIALIZED; default: throw new AssertionError(mError); } } private byte[] get(byte[] key) { ArrayList<byte[]> values = execute('g', key); return (values == null || values.isEmpty()) ? null : values.get(0); } public byte[] get(String key) { return get(getBytes(key)); } private boolean put(byte[] key, byte[] value) { execute('i', key, value); return mError == NO_ERROR; } public boolean put(String key, byte[] value) { return put(getBytes(key), value); } private boolean delete(byte[] key) { execute('d', key); return mError == NO_ERROR; } public boolean delete(String key) { return delete(getBytes(key)); } private boolean contains(byte[] key) { execute('e', key); return mError == NO_ERROR; } public boolean contains(String key) { return contains(getBytes(key)); } public byte[][] saw(byte[] prefix) { ArrayList<byte[]> values = execute('s', prefix); return (values == null) ? null : values .toArray(new byte[values.size()][]); } public String[] saw(String prefix) { byte[][] values = saw(getBytes(prefix)); if (values == null) { return null; } String[] strings = new String[values.length]; for (int i = 0; i < values.length; ++i) { strings[i] = toString(values[i]); } return strings; } public boolean reset() { execute('r'); return mError == NO_ERROR; } private boolean password(byte[] password) { execute('p', password); return mError == NO_ERROR; } public boolean password(String password) { return password(getBytes(password)); } public boolean lock() { execute('l'); return mError == NO_ERROR; } private boolean unlock(byte[] password) { execute('u', password); return mError == NO_ERROR; } public boolean unlock(String password) { return unlock(getBytes(password)); } public boolean isEmpty() { execute('z'); return mError == KEY_NOT_FOUND; } private boolean generate(byte[] key) { execute('a', key); return mError == NO_ERROR; } public boolean generate(String key) { return generate(getBytes(key)); } private boolean importKey(byte[] keyName, byte[] key) { execute('m', keyName, key); return mError == NO_ERROR; } public boolean importKey(String keyName, byte[] key) { return importKey(getBytes(keyName), key); } private byte[] getPubkey(byte[] key) { ArrayList<byte[]> values = execute('b', key); return (values == null || values.isEmpty()) ? null : values.get(0); } public byte[] getPubkey(String key) { return getPubkey(getBytes(key)); } private boolean delKey(byte[] key) { execute('k', key); return mError == NO_ERROR; } public boolean delKey(String key) { return delKey(getBytes(key)); } private byte[] sign(byte[] keyName, byte[] data) { final ArrayList<byte[]> values = execute('n', keyName, data); return (values == null || values.isEmpty()) ? null : values.get(0); } public byte[] sign(String key, byte[] data) { return sign(getBytes(key), data); } private boolean verify(byte[] keyName, byte[] data, byte[] signature) { execute('v', keyName, data, signature); return mError == NO_ERROR; } public boolean verify(String key, byte[] data, byte[] signature) { return verify(getBytes(key), data, signature); } private boolean grant(byte[] key, byte[] uid) { execute('x', key, uid); return mError == NO_ERROR; } public boolean grant(String key, int uid) { return grant(getBytes(key), Integer.toString(uid).getBytes()); } private boolean ungrant(byte[] key, byte[] uid) { execute('y', key, uid); return mError == NO_ERROR; } public boolean ungrant(String key, int uid) { return ungrant(getBytes(key), Integer.toString(uid).getBytes()); } public int getLastError() { return mError; } private ArrayList<byte[]> execute(int code, byte[]... parameters) { mError = PROTOCOL_ERROR; for (byte[] parameter : parameters) { if (parameter == null || parameter.length > 65535) { return null; } } LocalSocket socket = new LocalSocket(); try { socket.connect(sAddress); OutputStream out = socket.getOutputStream(); out.write(code); for (byte[] parameter : parameters) { out.write(parameter.length >> 8); out.write(parameter.length); out.write(parameter); } out.flush(); socket.shutdownOutput(); InputStream in = socket.getInputStream(); if ((code = in.read()) != NO_ERROR) { if (code != -1) { mError = code; } return null; } ArrayList<byte[]> values = new ArrayList<byte[]>(); while (true) { int i, j; if ((i = in.read()) == -1) { break; } if ((j = in.read()) == -1) { return null; } byte[] value = new byte[i << 8 | j]; for (i = 0; i < value.length; i += j) { if ((j = in.read(value, i, value.length - i)) == -1) { return null; } } values.add(value); } mError = NO_ERROR; return values; } catch (IOException e) { // ignore } finally { try { socket.close(); } catch (IOException e) { } } return null; } private static byte[] getBytes(String string) { try { return string.getBytes("UTF-8"); } catch (UnsupportedEncodingException e) { throw new RuntimeException(e); } } private static String toString(byte[] bytes) { try { return new String(bytes, "UTF-8"); } catch (UnsupportedEncodingException e) { throw new RuntimeException(e); } } }