【COSBench系列】4. COSBench 开发一个适配器 - sine-io/cosbench-sineio GitHub Wiki
COSBench系列的前三篇文章,讲解了它是什么为什么怎么用、开发环境、源码解读。在这最后一篇文章里,咱们来开发一个bundle(storage)以便后续进行SDK版本更新,代码逻辑修改等。
我们可以新建一个plug-in项目(选osgi),也可以选择一个偷懒的办法(选择一个近似的bundle进行拷贝后改造),这里我们选择后者。
a. 我们对cosbench-s3目录做一份拷贝,并进行重命名(我这里命名为了cosbench-news3)
b. 修改cosbench-news3目录中的 .project文件中的那么为cosbench-news3
c. eclipse中导入此bundle:File -> Import -> General -> Existing projects into Workspace -> Next,如下图所示
至此,新bundle改造完毕。
我们后续会升级aws-sdk-java的版本,因为升级后的SDK都升级了Java版本,所以咱在这里进行添加。Window -> Preferences -> Java -> Installed JREs -> Execution Environments里选择 JavaSE-1.8后,勾选Compatible JREs里的jre(我这里是1.8.0_301)
修改bundle的相关信息(ID等字段、Execution Environments里添加JavaSE-1.8)
建议新建个maven项目,然后使用maven进行相关jar包下载,我这里为了演示方便,直接把cosbench-news3直接转换为maven项目了。示例pom.xml如下(SDK版本可自行修改):
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<groupId>cosbench-news3</groupId>
<artifactId>cosbench-news3</artifactId>
<version>0.4.7.9</version>
<build>
<sourceDirectory>src</sourceDirectory>
<plugins>
<plugin>
<artifactId>maven-compiler-plugin</artifactId>
<version>3.8.1</version>
<configuration>
<source>1.6</source>
<target>1.6</target>
</configuration>
</plugin>
</plugins>
</build>
<dependencyManagement>
<dependencies>
<dependency>
<groupId>com.amazonaws</groupId>
<artifactId>aws-java-sdk-bom</artifactId>
<version>1.12.312</version>
<type>pom</type>
<scope>import</scope>
</dependency>
</dependencies>
</dependencyManagement>
<dependencies>
<dependency>
<groupId>com.amazonaws</groupId>
<artifactId>aws-java-sdk-s3</artifactId>
</dependency>
</dependencies>
</project>
当maven将依赖包下载完毕后,我们对照Maven Dependencies里的jar包路径,将其均拷贝到cosbench-news3目录下,并删掉之前的版本,结果如下:
对cosbench-news3项目进行右键刷新,然后修改META-INF里面的MANIFEST.MF文件,如下图所示:
此时,SDK升级完毕。
接下来,我们进行bundle开发。
对要进行修改的包或文件进行右键 -> Refactor -> Rename,修改成自定义的。我修改的如下图:
通过Eclipse,它会直接将类名也进行修改,接下来我们来看一下:
基础工作修改完毕,后续就是有需求后进行coding了。
大家记住NewS3Storage.java这个文件,我们经常与它打交道
我们使用以下三种需求,来依次讲解一下,这三个需求难度依次递增。
a. 添加配置参数
// NewS3Constants.java
// 添加下面两行配置
String NO_VERIFY_SSL_KEY = "no_verify_ssl";
boolean NO_VERIFY_SSL_DEFAULT = false;
b. 添加代码逻辑
// NewS3Storage.java
// init方法内部添加如下逻辑
boolean noVerifySSL = config.getBoolean(NO_VERIFY_SSL_KEY, NO_VERIFY_SSL_DEFAULT);
if (noVerifySSL) {
System.setProperty(SDKGlobalConfiguration.DISABLE_CERT_CHECKING_SYSTEM_PROPERTY, "true");
}
至此,需求1就开发完毕了,至于如何使用,请看章节3
这里只修改 init方法中提示过期的部分,其余的请自行参照官网进行修改(如:doesBucketExist ---> doesBucketExistV2)。
// init方法
// 删除下面三行
client = new AmazonS3Client(myCredentials, clientConf);
client.setEndpoint(endpoint);
client.setS3ClientOptions(new S3ClientOptions().withPathStyleAccess(pathStyleAccess));
// 添加下面几行代码
client = AmazonS3ClientBuilder.standard().withCredentials(new AWSStaticCredentialsProvider(myCredentials))
.withClientConfiguration(clientConf).withEndpointConfiguration(new EndpointConfiguration(endpoint, ""))
.withPathStyleAccessEnabled(pathStyleAccess).build();
至此,我们将SDK中过期的用法替换为了新的使用方法。
我们添加一个新接口,需要修改几个bundle,我们依次来进行修改。
cosbench-api
// StorageAPI.java
// 添加一个restore接口
/**
* Restore an object from a container.
*
* @param container - the name of a container.
* @param object - the name of an object to be restored.
* @param config - the configuration used for this operation.
*/
public void restoreObject(String container, String object, Config config);
// NoneStorage.java
// 对接口进行实现
@Override
public void restoreObject(String container, String object, Config config) {
if (logging) {
logger.info("performing RESTORE(restore-object) at /{}/{}", container, object);
}
}
cosbench-news3
// NewS3Constants.java
// 添加一个参数:取回天数
String RESTORE_DAYS_KEY = "restore_days";
int RESTORE_DAYS_DEFAULT = 1;
// NewS3Storage.java
// 0. 在添加公共参数
private AmazonS3 restoreClient;
private boolean pathStyleAccess;
private String proxyHost;
private String proxyPort;
private boolean noVerifySSL;
private int restoreDays;
// 1. 修改init方法如下
@Override
public void init(Config config, Logger logger) {
super.init(config, logger);
timeout = config.getInt(CONN_TIMEOUT_KEY, CONN_TIMEOUT_DEFAULT);
endpoint = config.get(ENDPOINT_KEY, ENDPOINT_DEFAULT);
accessKey = config.get(AUTH_USERNAME_KEY, AUTH_USERNAME_DEFAULT);
secretKey = config.get(AUTH_PASSWORD_KEY, AUTH_PASSWORD_DEFAULT);
pathStyleAccess = config.getBoolean(PATH_STYLE_ACCESS_KEY, PATH_STYLE_ACCESS_DEFAULT);
proxyHost = config.get(PROXY_HOST_KEY, "");
String proxyPort = config.get(PROXY_PORT_KEY, "");
parms.put(PROXY_PORT_KEY, proxyPort);
noVerifySSL = config.getBoolean(NO_VERIFY_SSL_KEY, NO_VERIFY_SSL_DEFAULT);
if (noVerifySSL) {
System.setProperty(SDKGlobalConfiguration.DISABLE_CERT_CHECKING_SYSTEM_PROPERTY, "true");
}
restoreDays = config.getInt(RESTORE_DAYS_KEY, RESTORE_DAYS_DEFAULT);
initClient();
initRestoreClient();
}
// 2. 在NewS3Storage类里面添加如下3个方法
private ClientConfiguration getDefaultClientConfiguration(String signType) {
ClientConfiguration defaultClientConfiguration = new ClientConfiguration();
// Set connection timeout for initially establishing a connection.
defaultClientConfiguration.setConnectionTimeout(timeout);
// Set socket timeout for data to be transferred.
defaultClientConfiguration.setSocketTimeout(timeout);
// use expect continue HTTP/1.1 header.
defaultClientConfiguration.withUseExpectContinue(false);
// Set Signer type.
defaultClientConfiguration.withSignerOverride(signType);
if ((!proxyHost.equals("")) && (!proxyPort.equals(""))) {
defaultClientConfiguration.setProxyHost(proxyHost);
defaultClientConfiguration.setProxyPort(parms.getInt(PROXY_PORT_KEY));
}
return defaultClientConfiguration;
}
private AmazonS3 initClient() {
logger.debug("initialize S3 client with storage config: {}", parms);
ClientConfiguration clientConf = getDefaultClientConfiguration("S3SignerType");
AWSCredentials myCredentials = new BasicAWSCredentials(accessKey, secretKey);
EndpointConfiguration myEndpoint = new EndpointConfiguration(endpoint, "");
client = AmazonS3ClientBuilder.standard().withCredentials(new AWSStaticCredentialsProvider(myCredentials))
.withClientConfiguration(clientConf).withEndpointConfiguration(myEndpoint)
.withPathStyleAccessEnabled(pathStyleAccess).build();
logger.debug("S3 client has been initialized");
return client;
}
private AmazonS3 initRestoreClient() {
logger.debug("initialize S3 Restore client with storage config: {}", parms);
ClientConfiguration clientConf = getDefaultClientConfiguration("AWSS3V4SignerType");
AWSCredentials myCredentials = new BasicAWSCredentials(accessKey, secretKey);
EndpointConfiguration myendpoint = new EndpointConfiguration(endpoint, "");
restoreClient = AmazonS3ClientBuilder.standard()
.withCredentials(new AWSStaticCredentialsProvider(myCredentials)).withClientConfiguration(clientConf)
.withEndpointConfiguration(myendpoint)
.withPathStyleAccessEnabled(pathStyleAccess).build();
logger.debug("S3 Restore client has been initialized");
return restoreClient;
}
// NewS3Storage.java
// 在NewS3Storage类中添加取回方法
@Override
public void restoreObject(String container, String object, Config config) {
super.restoreObject(container, object, config);
try {
RestoreObjectRequest request = new RestoreObjectRequest(container, object, restoreDays);
restoreClient.restoreObjectV2(request);
ObjectMetadata response = restoreClient.getObjectMetadata(container, object);
Boolean restoreFlag = response.getOngoingRestore();
logger.info(object + " at bucket -> " + container + " | Restore days: " + restoreDays
+ ", and ongoing-request status is: " + restoreFlag);
} catch (Exception e) {
throw new StorageException(e);
}
}
cosbench-driver
// Operators.java
// createOperator方法里添加restore判断
if (StringUtils.equals(type, Restorer.OP_TYPE)) {
return new Restorer();
}
// 在com.intel.cosbench.driver.operator包里新建一个java文件(Restorer.java)
// 内容如下:
package com.intel.cosbench.driver.operator;
import java.util.Date;
import com.intel.cosbench.bench.*;
import com.intel.cosbench.config.Config;
import com.intel.cosbench.driver.util.ObjectPicker;
import com.intel.cosbench.service.AbortedException;
import com.intel.cosbench.api.storage.StorageException;
import com.intel.cosbench.api.storage.StorageInterruptedException;
class Restorer extends AbstractOperator {
public static final String OP_TYPE = "restore";
private ObjectPicker objPicker = new ObjectPicker();
public Restorer() {
/* empty */
}
@Override
protected void init(String id, int ratio, String division, Config config) {
super.init(id, ratio, division, config);
objPicker.init(division, config);
}
@Override
public String getOpType() {
return OP_TYPE;
}
@Override
protected void operate(int idx, int all, Session session) {
String[] path = objPicker.pickObjPath(session.getRandom(), idx, all);
Sample sample = doRestoreV2(path[0], path[1], config, session, this);
session.getListener().onSampleCreated(sample);
Date now = sample.getTimestamp();
Result result = new Result(now, getId(), getOpType(), getSampleType(), getName(), sample.isSucc());
session.getListener().onOperationCompleted(result);
}
public static Sample doRestoreV2(String conName, String objName, Config config, Session session, Operator op) {
if (Thread.interrupted())
throw new AbortedException();
long start = System.nanoTime();
try {
session.getApi().restoreObject(conName, objName, config);
} catch (StorageInterruptedException sie) {
doLogErr(session.getLogger(), sie.getMessage(), sie);
throw new AbortedException();
} catch (StorageException se) {
String msg = "Error restore-object " + conName + "/" + objName + " " + se.getMessage();
doLogWarn(session.getLogger(), msg);
return new Sample(new Date(), op.getId(), op.getOpType(), op.getSampleType(), op.getName(), false);
} catch (Exception e) {
isUnauthorizedException(e, session);
errorStatisticsHandle(e, session, conName + "/" + objName);
return new Sample(new Date(), op.getId(), op.getOpType(), op.getSampleType(), op.getName(), false);
}
long end = System.nanoTime();
return new Sample(new Date(), op.getId(), op.getOpType(), op.getSampleType(), op.getName(), true,
(end - start) / 1000000, 0L, 0L);
}
}
至此,“需求3”就开发完毕,接下来在第三章节,咱们进行配置修改,修改完配置后,就可以打包进行验证了。
cd cosbench-sineio/release/conf/.driver
vi config.ini # 或者直接使用nodepad++等类似编辑器打开
# 添加如下配置:
plugins/cosbench-news3@7\:start,\
cd cosbench-sineio/release
vi start-driver.sh # 或者直接使用nodepad++等类似编辑器打开
# 大约45行,添加 cosbench-news3_${VERSION}
OSGI_BUNDLES="... cosbench-s3_${VERSION} cosbench-news3_${VERSION} ... cosbench-driver-web_${VERSION}"
打包流程请参考第二篇文章,这里按下不表,直接说部署新包后进行验证。
当我们将新版本部署好后,可以使用下面的xml配置来进行验证。
<?xml version="1.0" encoding="UTF-8" ?>
<workload name="s3 test" description="xxx">
<storage type="news3" config="endpoint=https://ip:port;accesskey=xxx;secretkey=xxx;no_verify_ssl=true;restore_days=1;path_style_access=true;timeout=60000" />
<workflow>
<workstage name="create bucket">
<work type="init" workers="1" config="cprefix=bkt0;containers=r(1,1)" />
</workstage>
<workstage name="prepare example">
<work type="prepare" workers="1" config="cprefix=bkt0;containers=r(1,1);objects=r(1,10);sizes=c(64)KB" />
</workstage>
<workstage name="restore object example">
<work name="restore test" workers="5" totalOps="10">
<operation type="restore" ratio="100" config="cprefix=bkt0;containers=c(1);objects=r(1,10)" />
</work>
</workstage>
</workflow>
</workload>
注意以下几点:
- storage的type修改为news3
- storage的config里添加配置 no_verify_ssl和restore_days
这样就会使用咱们新建的storage,并且使用咱们配置的参数
第四篇是终章,本系列到了说再见的时候了。
这系列文章难度逐步递进,我把它整理出来,供大家参考。
如果可以,欢迎在GitHub上给我的项目来个Star ⭐ ,这将给予我极大的动力和精神鼓励。也非常欢迎提交PR和issue,谢谢。
朋友们,江湖再见。:seedling: