Joget DX 8 Stable Released
The stable release for Joget DX 8 is now available, with a focus on UX and Governance.
在本教程中,我们将遵循开发插件的 指南 来开发我们的Amazon S3 Datalist Binder插件。 有关更多详细信息步骤,请参阅第一个教程 如何开发一个Bean Shell哈希变量插件。
我们要检索Amazon S3中的文件信息。
我们可以为此开发一个Datalist Binder插件或用户界面菜单插件。在本教程中,我们将开发一个 Datalist Binder插件 来检索文件信息并使用列表设计器填充它。
Datalist Binder不适合用于此目的的插件类型,因为Amazon S3客户端API无法获取文件的总数,也无法支持排序,分页和过滤等数据列表操作。我们这样写是为了学习的目的,不鼓励生产使用,因为它会有性能问题。
更好的方法是开发一个用户视图菜单,该菜单可以将文件显示为树结构,并在扩展树时加载额外的文件。
要开发Amazon S3 Datalist Binder插件,我们需要提供如下输入:
我们将在这里做一些不同的事情,因为一些输入将被放置在属性文件中,并在需要时从属性文件中检索。请参考如何在 文件上传表单元素与Amazon S3集成。
一个数据列表,它将根据配置列出Amazon S3存储桶中的文件。
我们需要始终准备好我们的Joget工作流程源代码,并按照这个指导方针建立起来 。
下面的教程是用Macbook Pro编写的,Joget源代码是5.0.1版本。 其他平台命令请参阅 如何开发插件文章。
假设我们的文件夹目录如下所示。
- Home - joget - plugins - jw-community -5.0.1
“插件”目录是我们将创建和存储我们所有插件的文件夹,“jw-community”目录是Joget Workflow源代码的存储位置。
运行以下命令在“plugins”目录下创建一个maven项目。
cd joget/plugins/ ~/joget/jw-community/5.0.1/wflow-plugin-archetype/create-plugin.sh org.joget amazon_s3_datalist_binder 5.0.1
然后,shell脚本会要求我们输入插件的版本号,并在生成maven项目之前要求我们确认。
Define value for property 'version': 1.0-SNAPSHOT: : 5.0.0 [INFO] Using property: package = org.joget Confirm properties configuration: groupId: org.joget artifactId: amazon_s3_datalist_binder version: 5.0.0 package: org.joget Y: : y
我们应该在终端上显示“BUILD SUCCESS”消息,并在“plugins”文件夹中创建一个“amazon_s3_datalist_binder”文件夹。
用你最喜欢的IDE打开maven项目。我将使用 NetBeans。
在“org.joget”包下创建一个“AmazonS3DatalistBinder”类。然后,使用org.joget.apps.datalist.model.DataListBinderDefault 抽象类来扩展 该类。请参阅 Datalist活页夹插件。我们还需要实现 org.joget.plugin.base.PluginWebSupport 接口类,以在插件属性页面中提供Ajax验证。请参考 Web服务插件。
像往常一样,我们必须执行所有的抽象方法。我们将使用AppPluginUtil.getMessage方法来支持i18n,并使用常量变量MESSAGE_PATH作为消息资源包目录。
现在,我们必须为管理员用户创建一个UI,为我们的插件提供输入。在getPropertyOptions方法中,我们已经指定了我们的 插件属性选项和配置 定义文件位于“/properties/amazonS3DatalistBinder.json”。让我们在“amazon_s3_datalist_binder / src / main”目录下创建一个目录“resources / properties”。创建目录后,在“properties”文件夹中创建一个名为“amazonS3DatalistBinder.json”的文件。
在属性定义选项文件中,我们需要提供如下的选项。请注意,我们可以在我们的属性选项中使用“@@ message.key @@”语法来支持i18n。如前所述,某些属性将放入一个属性文件中,因此我们的插件属性选项和配置 定义文件中只存在一个 属性。我们将通过AJAX验证来验证属性文件是否存在并能够连接到Amazon S3服务。
[{ title : '@@AmazonS3DatalistBinder.config@@', properties : [{ name : 'folder', label : '@@AmazonS3DatalistBinder.folder@@', type : 'textfield' }], validators : [{ type : 'AJAX', url : '[CONTEXT_PATH]/web/json/app[APP_PATH]/plugin/org.joget.AmazonS3DatalistBinder/service?action=validate' }] }]
其他属性将放在 awsS3.properties 文件中。这个属性文件将需要放在您的wflow文件夹中。
完成收集输入的属性选项后,我们可以使用getColumns,getPrimaryKeyColumnName,getData和getDataTotalRowCount方法来处理插件的主要方法。
protected static AmazonS3 s3; protected static Properties properties; protected static DataListColumn[] columns; protected List<Map<String, Object>> cacheData; protected static String getPropertiesPath() { return SetupManager.getBaseSharedDirectory() + "awsS3.properties"; } public static AmazonS3 getClient() throws Exception { if (s3 == null) { FileInputStream fis = null; try { properties = new Properties(); fis = new FileInputStream(new File(getPropertiesPath())); properties.load(fis); BasicAWSCredentials awsCreds = new BasicAWSCredentials(properties.getProperty("access_key_id"), properties.getProperty("secret_access_key")); s3 = new AmazonS3Client(awsCreds); Region region = Region.getRegion(Regions.fromName(properties.getProperty("region"))); s3.setRegion(region); if (!s3.doesBucketExist(properties.getProperty("bucket"))) { Bucket bucket = s3.createBucket(properties.getProperty("bucket")); if (bucket == null) { throw new RuntimeException(AppPluginUtil.getMessage("AmazonS3DatalistBinder.bucketFilToCreate", AmazonS3DatalistBinder.class.getName(), MESSAGE_PATH)); } } } catch (Exception e) { LogUtil.error(AmazonS3DatalistBinder.class.getName(), e, ""); s3 = null; if (e instanceof FileNotFoundException) { throw new RuntimeException(AppPluginUtil.getMessage("AmazonS3DatalistBinder.configureationFileIsMissing", AmazonS3DatalistBinder.class.getName(), MESSAGE_PATH)); } else { throw e; } } finally { try { if (fis != null) { fis.close(); } } catch (Exception e) {} } } return s3; } protected List<Map<String, Object>> getCacheData() { if (cacheData == null) { cacheData = new ArrayList<Map<String, Object>>(); try { AmazonS3 client = getClient(); String prefix = getPropertyString("folder"); if (prefix.isEmpty()) { prefix = null; } ObjectListing listing = client.listObjects(properties.getProperty("bucket"), prefix); boolean cont; do { cont = false; List<S3ObjectSummary> summaries = listing.getObjectSummaries(); for (S3ObjectSummary s : summaries) { Map<String, Object> obj = new HashMap<String, Object>(); String key = s.getKey(); int pos = key.lastIndexOf("/"); obj.put("key", key); obj.put("path", (pos > 0)?(key.substring(0, pos-1)):""); obj.put("filename", (pos > 0)?(key.substring(pos+1)):key); obj.put("owner", s.getOwner().getDisplayName()); obj.put("md5", s.getETag()); obj.put("size", s.getSize()); obj.put("storageClass", s.getStorageClass()); obj.put("lastModified", s.getLastModified()); cacheData.add(obj); } if (listing.isTruncated()) { cont = true; listing = client.listNextBatchOfObjects(listing); } } while (cont); } catch (Exception e) {} } return cacheData; } public DataListColumn[] getColumns() { if (columns == null) { Collection<DataListColumn> list = new ArrayList<DataListColumn>(); list.add(new DataListColumn("key", AppPluginUtil.getMessage("AmazonS3DatalistBinder.key", getClassName(), MESSAGE_PATH), true)); list.add(new DataListColumn("path", AppPluginUtil.getMessage("AmazonS3DatalistBinder.path", getClassName(), MESSAGE_PATH), true)); list.add(new DataListColumn("filename", AppPluginUtil.getMessage("AmazonS3DatalistBinder.filename", getClassName(), MESSAGE_PATH), true)); list.add(new DataListColumn("owner", AppPluginUtil.getMessage("AmazonS3DatalistBinder.owner", getClassName(), MESSAGE_PATH), true)); list.add(new DataListColumn("md5", AppPluginUtil.getMessage("AmazonS3DatalistBinder.md5", getClassName(), MESSAGE_PATH), false)); list.add(new DataListColumn("size", AppPluginUtil.getMessage("AmazonS3DatalistBinder.size", getClassName(), MESSAGE_PATH), true)); list.add(new DataListColumn("storageClass", AppPluginUtil.getMessage("AmazonS3DatalistBinder.storageClass", getClassName(), MESSAGE_PATH), true)); list.add(new DataListColumn("lastModified", AppPluginUtil.getMessage("AmazonS3DatalistBinder.lastModified", getClassName(), MESSAGE_PATH), true)); columns = list.toArray(new DataListColumn[]{}); } return columns; } public String getPrimaryKeyColumnName() { return "key"; } public DataListCollection getData(DataList dataList, Map properties, DataListFilterQueryObject[] filterQueryObjects, String sort, Boolean desc, Integer start, Integer rows) { //TODO: handle filterQueryObjects List list = PagingUtils.sortAndPage(getCacheData(), sort, desc, start, rows); DataListCollection data = new DataListCollection(); data.addAll(list); return data; } public int getDataTotalRowCount(DataList dataList, Map properties, DataListFilterQueryObject[] filterQueryObjects) { //TODO: handle filterQueryObjects return getCacheData().size(); }
在我们的插件属性中,我们有一个AJAX验证来测试属性文件和连接到Amazon S3客户端。让我们实现webService方法来提供一个用于验证的API。
public void webService(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { boolean isAdmin = WorkflowUtil.isCurrentUserInRole(WorkflowUserManager.ROLE_ADMIN); if (!isAdmin) { response.sendError(HttpServletResponse.SC_UNAUTHORIZED); return; } String action = request.getParameter("action"); if ("validate".equals(action)) { String message = ""; boolean success = true; try { AmazonS3DatalistBinder.getClient(); } catch (Exception e) { LogUtil.error(this.getClassName(), e, ""); success = false; message = StringUtil.escapeString(e.getMessage(), StringUtil.TYPE_JAVASCIPT, null); } try { JSONObject jsonObject = new JSONObject(); jsonObject.accumulate("status", (success?"success":"fail")); JSONArray messageArr = new JSONArray(); messageArr.put(message); jsonObject.put("message", messageArr); jsonObject.write(response.getWriter()); } catch (Exception e) { //ignore } } else { response.setStatus(HttpServletResponse.SC_NO_CONTENT); } }
我们需要在我们的POM文件中包含“jsp-api”和“aws-java-sdk-s3”库。
<!-- Change plugin specific dependencies here --> <dependency> <groupId>javax.servlet</groupId> <artifactId>jsp-api</artifactId> <version>2.0</version> </dependency> <dependency> <groupId>com.amazonaws</groupId> <artifactId>aws-java-sdk-s3</artifactId> <version>1.10.56</version> </dependency> <!-- End change plugin specific dependencies here -->
我们在getLabel和getDescription方法中使用i18n消息密钥。我们将在我们的属性选项定义中使用i18n消息密钥。然后,我们将需要为我们的插件创建一个消息资源包属性文件。
在“amazon_s3_datalist_binder / src / main”目录下创建一个目录“resources / message”。然后,在该文件夹中创建一个“AmazonS3DatalistBinder.properties”文件。在属性文件中,添加所有消息密钥及其标签,如下所示。
org.joget.AmazonS3DatalistBinder.pluginLabel=Amazon S3 Datalist Binder org.joget.AmazonS3DatalistBinder.pluginDesc=Used to retrieve the available files in Amazon S3. AmazonS3DatalistBinder.config=Configure Amazon S3 Datalist Binder AmazonS3DatalistBinder.configureationFileIsMissing=AWS S3 configuration file is missing. AmazonS3DatalistBinder.bucketFilToCreate=AWS Bucket fail to create. AmazonS3DatalistBinder.key=Key AmazonS3DatalistBinder.path=Path AmazonS3DatalistBinder.filename=File Name AmazonS3DatalistBinder.owner=Owner AmazonS3DatalistBinder.md5=MD5 Hash AmazonS3DatalistBinder.size=Size AmazonS3DatalistBinder.storageClass=Storage Class AmazonS3DatalistBinder.lastModified=Last Modified AmazonS3DatalistBinder.folder=Folder
接下来,我们将需要在Activator类(在同一个类包中自动生成)中注册我们的插件类,以告诉Felix框架这是一个插件。
public void start(BundleContext context) { registrationList = new ArrayList<ServiceRegistration>(); //Register plugin here registrationList.add(context.registerService(AmazonS3DatalistBinder.class.getName(), new AmazonS3DatalistBinder(), null)); }
让我们建立我们的插件。构建过程完成后,我们将在“amazon_s3_datalist_binder / target”目录下找到一个“amazon_s3_datalist_binder-5.0.0.jar”文件。
然后,让我们上传插件jar到 管理插件。上传jar文件后,再次检查插件是否正确上传并激活。
检查Amazon S3 Datalist Binder插件可在 列表设计器中找到。
选择并配置Amazon S3 Datalist活页夹。
按下确定。它的awsS3.properties 属性文件丢失或无效。将显示错误消息。
设计数据表。
检查结果。
您可以从amazon_s3_datalist_binder_src.zip下载源代码 。