一个基于Key/Value方式存取二进制数据的接口设计

  • 支持对单个条目进行加密
  • 增删改查方法
  • 辅助方法
  • 能够使用以下5种方式进行序列化
    • "SQLITE" Sqlite数据库
    • "XML" XML文件
    • "INI" INI文件
    • "DIR" 文件系统(目录结构)
    • "REG" Windows注册表
  • 支持拷贝
  • 完整的错误处理
  • 支持尽可能多的STL容器类型作为buf
  • 线程安全(可选)

C++

要求能通过不低于此测试用例强度的单元测试:http://git.koal.com/chenlei/blobstore/blob/master/blobstore/blobstore_test.cc

BlobStore 使用范例(C++伪代码)


XMLBlobStore xml_store("/path/to/xml"); char in_buf1[] = "11111111111111111111111111";
std::string in_buf2 = "222222222222222222222222";
std::vector<char> in_buf3(in_buf1, in_buf1 + sizeof(in_buf1)); // 设置明文
xml_store.setBlob("testkey1", in_buf1, sizeof(in_buf1));
xml_store.setBlob("testkey2", in_buf2);
xml_store.setBlob("testkey3", in_buf3)
// 以明文方式取出
char out_buf1[MAX];
std::string out_buf2;
std::vector<char> outbuf2;
xml_store.getBlob("testkey1", out_buf1, sizeof(out_buf1));
xml_store.getBlob("testkey2", out_buf2);
xml_store.getBlob("testkey2", out_buf3); // 设置密文
xml_store.setEncryptedBlob("testkey1", passwd, in_buf1, sizeof(in_buf1));
xml_store.setBlob("testkey2", passwd, in_buf2, in_buf2.size());
// 以密文方式取出
char out_buf1[MAX];
std::string out_buf2;
xml_store.getBlob("testkey1", passwd, out_buf1, sizeof(out_buf1));
xml_store.getBlob("testkey2", passwd, out_buf2); // 辅助方法
std::vector<std::string> keys = xml_store.keys();
bool has_key = xml_store.hasKey("testkey1");
bool is_encryped = xml_store.isEncrypted("testkey1"); // 序列化
xml_store.load(); // 从XML文件加载
xml_store.save(); // 序列化到XML文件 // 拷贝
XMLBlobStore new_xml_store("/path/to/new/xml");
blobstore_copy(new_xml_store, xml_store);
new_xml_store.save(); DBBlobStore db_store("/path/to/db");
blobstore_copy(db_store, xml_store);
db_store.save();

一个可能的接口设计示例(C++)

//一个基于Key/Value方式存取二进制数据的接口设计,支持对单个条目进行加密

/* 支持加密的二进制数据存储 */
class BlobStore {
public:
/**
* 构造器
* @param type,支持以下4种
* "DIR" 文件系统(目录结构)
* "REG" Windows注册表
* "SQLITE" Sqlite数据库
* "XML" XML文件
* @param storePath,对应不同type的路径定义如下
* "DIR" 文件目录路径
* "REG" Windows注册表路径
* "SQLITE" Sqlite数据库文件路径
* "XML" XML文件路径
*/
static BlobStore createInstance(string type, string storePath); string getType(); /* 持久化 */
int load();
int save(); /* 将所有Key/Value复制到另一个Store(可以是不同类型的Store) */
int copyTo(BlobStore &other); public:
/* 遍历与查找 */
int size(); //获取容量
set<string> listAliases(); //获取名称列表
boolean containsAlias(string alias); //判断存在性
boolean isEncrypted(string alias); //判断条目是否加密
string findAlias(const vector<BYTE>&); //根据二进制内容查找名称 /* 明文数据操作 */
int getBlob(string alias, vector<BYTE> & );
int setBlob(string alias, const vector<BYTE>& ); /* 密文数据操作(固定使用SM4或AES256-CBC加密,密钥使用password的MD5摘要值) */
int getEncryptedBlob(string alias, string entryPassword, vector<BYTE> &);
int setEncryptedBlob(string alias, string entryPassword, const vector<BYTE>&); /* 删除 */
int deleteBlob(string alias);
int clearAll();
}

Java

要求使用JUNIT对以下所有接口进行单元测试(包括异常流)

BlobStore 使用范例(Java伪代码)


IBlobStore xs = BlobStoreFactory.createInstance("XML", "/path/to/xml"); String in1 = "11111111111111111111111111";
byte[] in2 = {0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08}; String passwd = "abcdefgh"; // 设置明文
xs.setBlob("testkey1", in1);
xs.setBlob("testkey2", in2); String out1 = xs.getBlobAsString("testkey1");
byte[] out2 = xs.getBlob("testkey2"); // 设置密文
xs.setEncryptedBlob("testkey1", passwd, in1);
xs.setEncryptedBlob("testkey2", passwd, in2); // 以密文方式取出
out1 = xs.getEncryptedBlobAsString("testkey1", passwd);
out2 = xs.getEncryptedBlob("testkey2", passwd); // 辅助方法
String[] keys = xs.listAliases();
boolean hasKey = xs.containsAlias("testkey1");
boolean isEncrypted = xs.isEncrypted("testkey2"); // 序列化
xs.load(); // 从XML文件加载
xs.save(); // 序列化到XML文件 // 拷贝
IBlobStore xs2 = BlobStore.createInstance("XML", "/path/to/new_xml");
xs.copyTo(xs2);
xs2.save(); IBlobStore ds = BlobStore.createInstance("DB", "/path/to/test.db");
xs.copyTo(xs2);
xs2.save();

一个可能的接口设计示例(Java)

interface IBlobStore {
public String getType(); /* 持久化 */
public int load();
public int save(); /* 将所有Key/Value复制到另一个Store(可以是不同类型的Store) */
public int copyTo(IBlobStore other); /* 遍历与查找 */
public int size(); //获取容量
public String[] listAliases(); //获取名称列表
public boolean containsAlias(String alias); //判断存在性
public boolean isEncrypted(String alias); //判断条目是否加密
public String findAlias(byte[] value); //根据二进制内容查找名称 /* 明文数据操作 */
public int setBlob(String alias, String value);
public int setBlob(String alias, byte[] value);
public byte[] getBlob(String alias);
public String getBlobAsString(String alias); /* 密文数据操作(固定使用SM4或AES256-CBC加密,密钥使用password的MD5摘要值) */
public int setEncryptedBlob(String alias, String entryPassword, String value);
public int setEncryptedBlob(String alias, String entryPassword, byte[] value);
public byte[] getEncryptedBlob(String alias, String entryPassword);
public String getEncryptedBlobAsString(String alias, String entryPassword); /* 删除 */
int deleteBlob(String alias);
int clearAll();
} class BlobStoreFactory { /**
* 构造器
* @param type,支持以下4种
* "DIR" 文件系统(目录结构)
* "REG" Windows注册表
* "SQLITE" Sqlite数据库
* "XML" XML文件
* @param storePath,对应不同type的路径定义如下
* "DIR" 文件目录路径
* "REG" Windows注册表路径
* "SQLITE" Sqlite数据库文件路径
* "XML" XML文件路径
*/
static IBlobStore createInstance(String type, String storePath) {
...
}
} class XMLBlobStore implements IBlobStore {
...
} class DbBlobStore implements IBlobStore {
...
}

Practice encryptedblobstore的更多相关文章

  1. Pramp mock interview (4th practice): Matrix Spiral Print

    March 16, 2016 Problem statement:Given a 2D array (matrix) named M, print all items of M in a spiral ...

  2. Atitit 数据存储视图的最佳实际best practice attilax总结

    Atitit 数据存储视图的最佳实际best practice attilax总结 1.1. 视图优点:可读性的提升1 1.2. 结论  本着可读性优先于性能的原则,面向人类编程优先于面向机器编程,应 ...

  3. The Practice of .NET Cross-Platforms

    0x01 Preface This post is mainly to share the technologies on my practice about the .NET Cross-Platf ...

  4. Exercise 24: More Practice

    puts "Let's practice everything." puts 'You\'d need to know \'bout escapes with \\ that do ...

  5. ConCurrent in Practice小记 (3)

    ConCurrent in Practice小记 (3) 高级同步技巧 Semaphore Semaphore信号量,据说是Dijkstra大神发明的.内部维护一个许可集(Permits Set),用 ...

  6. ConCurrent in Practice小记 (2)

    Java-ConCurrent2.html :first-child{margin-top:0!important}img.plugin{box-shadow:0 1px 3px rgba(0,0,0 ...

  7. ConCurrent in Practice小记 (1)

    ConCurrent in Practice小记 (1) 杂记,随书自己写的笔记: 综述问题 1.线程允许在同一个进程中的资源,包括共享内存,内存句柄,文件句柄.但是每个进程有自己的程序计数器,栈和局 ...

  8. 1.2 基础知识——关于猪皮(GP,Generic Practice)

    摘要: 这是<CMMI快乐之旅>系列文章之一.说起猪皮(GP,Generic Practice),真的让人又爱又恨,中文翻译叫通用实践.CMMI标准中每个级别包含几个PA,每个PA又包含几 ...

  9. 2015年第2本(英文第1本):《The Practice of Programming》

    2015年计划透析10本英文原著,最开始选定的第一本英文书是<Who Moved my Cheese>,可是这本书实在是太短.太简单了,总体的意思就是要顺应变化,要跳出自己的舒适区,全文不 ...

随机推荐

  1. 从生成文件对比两种创建虚拟机的方式:boot from image和boot from bootable-volume

    1. 创建bootable-volume(参考:http://docs.openstack.org/grizzly/openstack-compute/admin/content/instance-c ...

  2. Java探索之旅(10)——数组线性表ArrayList和字符串生成器StringBuffer/StringBuilder

    1.数组线性表ArrayList 数组一旦定义则不可改变大小.ArrayList可以不限定个数的存储对象.添加,插入,删除,查找比较数组更加容易.可以直接使用引用类型变量名输出,相当于toString ...

  3. fdisk查看硬盘分区表

    fdisk [选项] <磁盘>    更改分区表 fdisk [选项] -l <磁盘> 列出分区表 fdisk -s <分区>        给出分区大小(块数) ...

  4. [51nod1043]幸运号码

    题意:1个长度为2N的数,如果左边N个数的和 = 右边N个数的和,那么就是一个幸运号码. 例如:99.1230.123312是幸运号码. 给出一个N,求长度为2N的幸运号码的数量.由于数量很大,输出数 ...

  5. linux下将编译错误输出到一个文本文件

    linux下将编译错误输出到一个文本文件 command > filename 把把标准输出重定向到一个新文件中 command > > filename 把把标准输出重定向到一个文 ...

  6. R: data.frame 生成、操作数组。重命名、增、删、改

    ################################################### 问题:生成.操作数据框   18.4.27 怎么生成数据框 data.frame.,,及其相关操 ...

  7. Fiddler开启Https的时候出现unable to configure windows to trust Fiddler Root certificate问题

    前言 通过log页面看到错误为:访问控制列表(ACL)结构无效. 网上(baidu,bing,google)各种方式都试过了: 如重置证书(Reset all certificates) 导出证书到本 ...

  8. 在Android中使用Protocol Buffers(下篇)

    本文来自网易云社区. FlatBuffers编码数组 编码数组的过程如下: 先执行 startVector(),这个方法会记录数组的长度,处理元素的对齐,准备足够的空间,并设置nested,用于指示记 ...

  9. PHP中的继承

    <?php class Bar { private $salary = 3000; public $lunch = 1000; // php中关于“可见性”的概念 public function ...

  10. unity 3d 之合并网格和贴图(combine mesh and texture)

    https://www.cnblogs.com/eangulee/p/3877824.html unity 3d 之合并网格和贴图(combine mesh and texture)   本人是个小白 ...