terça-feira, 5 de maio de 2020

Using AntRun to run a Maven plugin on multiple files (if the plugin only allows 1 file at a time)


So, i was dealing with a plugin the other day that did not allow multiple declaration of files. Then i figured that Maven has a Maven run, that allows us to code a script to do it.
<properties>
...
    <joynr.generator.fidl.file>placeholder</joynr.generator.fidl.file>
...
</properties>
<plugins>
 ...
  <plugin>
    <groupId>io.joynr.tools.generator</groupId>
    <artifactId>joynr-generator-maven-plugin</artifactId>
    <version>${joynr.version}</version>

    <configuration>
      <model>${joynr.generator.fidl.file}</model>
      <generationLanguage>java</generationLanguage>
      <outputPath>${basedir}/src/main/generated-java</outputPath>
      <addVersionTo>package</addVersionTo>
    </configuration>
    
    <dependencies>
      <dependency>
        <groupId>io.joynr.tools.generator</groupId>
        <artifactId>java-generator</artifactId>
        <version>${joynr.version}</version>
      </dependency>
    </dependencies>
  </plugin>

  <plugin>
    <artifactId>maven-antrun-plugin</artifactId>
    <version>3.0.0</version>

    <executions>
      <execution>
        <phase>generate-sources</phase>
        <configuration>
          <target name="Generate Joynr Java artifacts from FIDL files">
            <apply executable="mvn" parallel="false">
              <arg value="io.joynr.tools.generator:joynr-generator-maven-plugin:generate" />
              <srcfile prefix="-Djoynr.generator.fidl.file=" />
              <fileset dir="src/main/resources">
                <patternset>
                  <include name="**/*.fidl" />
                </patternset>
              </fileset>
            </apply>
          </target>
        </configuration>
        <goals>
          <goal>run</goal>
        </goals>
      </execution>
    </executions>
  </plugin>
 ...
</plugins>

Let me explain a bit of what is happening in the above code: I'm declaring a variable that can be set from the command line environment using -Dvariable=value. Then i proceed to configure all the fixed parameters to the target plugin (in this case joynr-generator-maven-plugin), and then, i configure the Ant Run Plugin to run Maven for each FIDL file that it finds on a particular folder with more advanced filters (include/exclude).

Happy coding!

terça-feira, 15 de janeiro de 2019

How to Mavenize a Visual Studio project


Since i was so used to work with Maven, due to past Java+Maven experiences, when i was asked to work in C# i felt something was amiss.

Here are my reasons:
  1. The capability of building the solution in the Windows command line (without the need to open Visual Studio IDE)
  2. Being able to build some parts of the project including dependencies
  3. Being able to run test suites during the build
  4. Provide an automated solution that any member of the team should be able to build/test/run with a single command
  5. Integrate the automated solution with Jenkins
I don't know if Microsoft has their own Continuous Integration server/service, but i managed to come up with a pretty elegant solution for what i needed using Open Source/Publicly available software such as Maven/Jenkins, and so, i decided to share.

As usual, i will provide the sources of this project here, so you can download and inspect the sources for yourself, and adapt it to your needs.

Here is a recipe of how to build a new project from the ground up, as well as a showcase of what can be achieved using this set of technologies.

First off, lets create a new Solution/Project. In this case, i've created a WebAPI 2.0 project plus a test project that will be used to test if the WebServices are on-line or not (can also be used to test if the WebServices are working as expected)

This is a pretty standard process, and i will not go into the details on how to create this. I just used the Visual Studio wizards to help me create these 2 projects.

As you can see there is a HealthInfoController and a HealthInfoControllerTest. The first is the WebService we want to test, and the latter is the test. Note that, the test project does not contain any reference to the WebApp project. This means the WebApp.Tests build will not trigger a build of WebApp on the VisualStudio IDE, but it also means that the WebApp.Tests has no access to the WebApp internals, and cannot access the WebApp namespace.

The WebApp.Tests is therefore, an Integration Test project, not a Unit Test project. This means, we have to run WebApp on the IIS Server before starting the test suite.

After you have a buildable solution and tests, we need to add Maven artifacts to the solution, in order to be able to run this as a Maven project.

So, we need a file with instructions on how to start the IIS Server, called iis_express_start.bat. A pre-integration and post-integration scripts that will be used to launch and shutdown the IIS Server running as a child process. And we need a pom.xml and corresponding hierarchy in order to orchestrate this project as an IIS integration test project for Maven.


The Maven artifacts in the root directory include the following:
  • pom.xml - Root description of the Mavenized project
  • module.xml - Skeleton POM descriptor for each of the projects that we want to build and produce binary libraries (DLL's)
  • packageManager.xml - Maven descriptor for NuGet package management resolution
  • test_module.xml - skeleton POM descriptor for unit tests
  • test_module_iis.xml - skeleton POM descriptor for IIS server integration tests
As you can see, all the build + tests can be done from Maven, without the need to start Visual Studio. I can even edit the files using notepad if i'm in a rush.

Here is what a build of this project looks like:







Happy coding!

sexta-feira, 3 de março de 2017

Bashttpd - Yes! an HTTP daemon written for bash!

Last week i received a request for a web page prototype for an embedded system that i'm working on. Not wanting to go through the painstaking procedure of compiling all artifacts for a full fledged HTTPd daemon such as Lighttpd server, and because it is a prototype, it will be run in a controlled environment... I decided to say: hey, perhaps i could do it in bash! Why not? I just have to parse the request directory/file, and serve the file to the user...

I've started developing, but then it hit me: Why do it before looking for it? I can't be the only one thinking about this. And i googled this: https://github.com/avleen/bashttpd

Behold 'The holy grail'!

I've started investigating the sources and decided to try it out. Turns out, it is quite impressive what it can do and the speed it does for 'development' environments. Sure i don't recommend this for production environments, but if you ever have to do a prototype and have limited resources, look no further: This is a powerful candidate!

Of course that 'With great power comes great responsibility': You can do a lot of powerful stuff with this, but beware: put it in an unsafe environment and a 'monster' will be unleashed! Hackers can do all kinds of nasty stuff with it :D

Here is my fork: https://github.com/rochajoel/bashttpd

Happy Coding!

quarta-feira, 21 de dezembro de 2016

RESTful WebServices Tutorial with Maven+Tomcat+Failsafe

Technology credits:
Apache Maven & Failsafe
Apache Tomcat
Jersey



Disclosure

Before you start reading further: Keep in mind that this is a work in progress, and although i'm a "Systems&Informatics Engineer" i consider myself a novice programmer. If you have any question you don't see answered here, please leave your comment on the section below to help me improve this article.
Also, you should be aware that this post was a result of a lot of work. If you like it, please feel free to show your appreciation by offering me a cup of coffee through paypal and/or add value to this article by sharing your knowledge.

Objectives of this tutorial:

  • To establish a standard "archetype" for RESTful JAX-WS WebServices development
  • To demonstrate how a Maven project can build-deploy-test a RESTful WebService
Pre-requisites:
  • You have knowledge on Plain Old Java Objects (POJO's) development.
  • You understand the concept of WebService, it's benefits and are able to decide when and how it should be used.
  • You understand the diference between a RESTful WebService and a SOAP WebService
  • You understand the concept of "server side container" and it's purpose.
  • You have downloaded and installed the Maven build software, or you are using Maven through the m2e Eclipse plug-in
The sources for this project are available here. Just unzip it and build it using 'mvn clean verify' (on the cmd line or 'clean verify' targets using the m2e Eclipse plugin)

Conventions over Configuration

First of all, as we will be using the Maven WAR plugin we should be aware of the project's file structure that is required by this plugin (as well as other conventions required by Maven which i won't discuss here)
As you can see from the picture above, all you need to build and deploy a RESTful WebService to Tomcat are this 4 files (the ones above with the SVN icon). Be aware that the Maven WAR plugin requires you to have the web.xml at that specific location so that they can be included in the deployment to Tomcat. We will talk about each file in the following sections.

What should the pom.xml contain?

The pom.xml is the Maven build file and contains all the required configurations to build, deploy and test the project. In this tutorial, it only contains the configurations that are absolutely essential for this project to pass the tests. I will talk here about the essentials of the configuration that prepare the WAR for proper Tomcat deployment and the Failsafe plug-in.

Get ready for WAR

So... What do we need to get started? First of all let's start by calling some friends to the WAR :P We will need some "dependencies" to resolve our classpath needs:
1
2
3
4
5
6
7
8
9
<dependencies>
 ...
 <dependency>
  <groupId>com.sun.jersey</groupId>
  <artifactId>jersey-servlet</artifactId>
  <version>1.19.3</version>
 </dependency>
 ...
</dependencies>
This is the Jersey Servlet that will use our classes to generate URLs and forward the REST requests to the appropriate classes. This will be more clear when we configure the web.xml

Other than this, we just need to create the POJO files and annotate them with the appropriate declarations to be processed by Jersey

The WebService

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
@Path("/MyService")
public class MyService {
 @javax.ws.rs.GET
 @javax.ws.rs.Path(value="/myMethod")
 @javax.ws.rs.Produces(value={"application/json"})
 public JSONObject myMethod(
   @javax.ws.rs.QueryParam(value="name") String name) {
  JSONObject ret = new JSONObject();
   
  try {
   ret.put("greeting", "Hello " + name + "!");
  } catch (JSONException e) {
   e.printStackTrace();
  }
   
  return ret;
 }
}
In here i use a JSONObject return instead of a plain String so that this can be more than just the simple 'Hello World!' example. JSONObjects allow us to transfer complex datatypes over a standard data type language similar to XML but with a smaller footprint. This is great when you want to improve communications throughput or when working with constrained devices such as mobile phones or others. This also allows you to see that if you develop your own datatypes by extending from JSONObject you can return them like this and Jersey will take care of the serialization for you :)
Also: We can also receive JSONObjects as a parameter they work the same as a String, it's just a matter of URL encoding them on the request

That (aside from the web.xml configuration) will be all we need to run our first RESTful WebService
But don't trust me, let's test it using the Maven Failsafe&Tomcat plugins! :)

Integration test programmatically

To ensure the code above, and the integration of all technologies work as expected, lets develop a simple set of tests to ensure basic communication is as expected.

We will create tests started with IT*.java or ended in *IT.java or *ITCase.java. This classes are just like standard JUnit tests, only the filename pattern is different. The configuration is in the Maven POM but this time i wont go into details. You can see other tutorials that talk about this here
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
private String getResponseString(HttpURLConnection c) throws IOException {
 BufferedReader in = new BufferedReader( new InputStreamReader(c.getInputStream()) );
 String inputLine;
 StringBuffer responseString = new StringBuffer();
 
 while ((inputLine = in.readLine()) != null)
  responseString.append(inputLine);
 
 in.close();
  
 return responseString.toString();
}
 
@Test
public void testMyService_notNull() throws Exception {
 String myMethodArg0_Name = "rochajoel";
 String endpoint = "http://localhost:8091"
   + "/MyService"
   + "/myMethod";
 String parameters = "name=" + myMethodArg0_Name;
 
 HttpURLConnection c = (HttpURLConnection) new URL(endpoint + "?" + parameters).openConnection();
 
 final JSONObject response = new JSONObject(getResponseString(c));
  
 assertNotNull(response.get("greeting"));
 assertEquals("unexpected myMethod greeting",
   "Hello " + myMethodArg0_Name + "!",
   response.getString("greeting"));
}
 
@Test
public void testMyService_null() throws Exception {
 String myMethodArg0_Name = null;
 String endpoint = "http://localhost:8091"
   + "/MyService"
   + "/myMethod";
 
 HttpURLConnection c = (HttpURLConnection) new URL(endpoint).openConnection();
 
 final JSONObject response = new JSONObject(getResponseString(c));
  
 assertNotNull(response.get("greeting"));
 assertEquals("unexpected myMethod greeting",
   "Hello " + myMethodArg0_Name + "!",
   response.getString("greeting"));
}

The WebService configuration files

I acknowledge that it takes quite a configuration effort to put the WebService "up and running" but this is the last step. Take it easy :)

The web.xml:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
<?xml version="1.0" encoding="UTF-8"?>
<web-app>
 <display-name>MyService</display-name>
 
 <servlet>
  <servlet-name>restServlet</servlet-name>
  <servlet-class>com.sun.jersey.spi.container.servlet.ServletContainer
  </servlet-class>
 
  <init-param>
   <param-name>com.sun.jersey.config.property.packages</param-name>
   <param-value>ext.domain.pkg</param-value>
  </init-param>
  <load-on-startup>1</load-on-startup>
 </servlet>
 
 <servlet-mapping>
  <servlet-name>restServlet</servlet-name>
  <url-pattern>/*</url-pattern>
 </servlet-mapping>
 
 
 <session-config>
  <session-timeout>120</session-timeout>
 </session-config>
</web-app>
What are we saying here?
First of all, a WebService is not a Servlet. A servlet is the basic "entrance point" for any J2EE application. It's like a Main class for J2SE. Because of that, we must tell Tomcat which Servlet do we want him to invoke at what address. As we didn't develop any, we must use one from another provider :) This is were the com.sun.jersey.spi.container.servlet.ServletContainer comes along to save our back. Next, we configure it to inspect all classes in ext.domain.pkg and gather information from the annotations we provided earlier.
That's it! Just run your 'mvn clean verify' and you should be able to "succed your build" with 0 failures.

Happy Coding!

quinta-feira, 5 de fevereiro de 2015

Simple Maven archetype for Axis2 web services with Tomcat embedded

Technology credits:
Apache Maven & Failsafe
Apache Tomcat
Apache Axis2

Disclosure

Before you start reading further: Keep in mind that this is a work in progress, and although i'm a "Systems&Informatics Engineer" i consider myself a novice programmer. If you have any question you don't see answered here, please leave your comment on the section below to help me improve this article.
Also, you should be aware that this post was a result of a lot of work. If you like it, please feel free to show your appreciation by offering me a cup of coffee through paypal and/or add value to this article by sharing your knowledge.

Apache Axis2 architecture

If you check out some of my older posts, such as this you should by now be wondering whats the deal with Axis2. You can already deploy to Tomcat your own WebServices, so why bother have Axis2 on your WebServices?

Well... Axis2 deploys itself to your 'Application Server' (Tomcat or others) and only then will you be able to deploy your WebService in a special *.AAR format. This allows your 'Application Server Administrator' to deploy your WebApplication only once, give you the required privileges, and let you be responsible for upgrading/updating your own webservices.

This means (as you probably are guessing) that on the Maven building environment, there must be directives to:
  • Build an AAR instead of a WAR
  • Deploy the Axis2.WAR on Tomcat
  • Deploy your AAR on the Axis2.WAR WebApplication


The Maven Project

I've finally sum up the time to compile a simple but fully functional Maven Project for this post and is available here for your convenience.

Project Structure


The main differences from previous tutorials are on the Java implementation class (notice i've used my_method instead of myMethod for the method name, this was not an arbitrary choice, but a consequence of a drawback of Axis2).
Second, the web.xml reflects the fact that we'll be using the Axis2 container instead of a standalone container.
And third: the deep nesting of folders to define the services.xml configuration file under WEB-INF folder.
Everything else is pretty much the same as in this tutorial.

Since you have access to the project's source code, I'll leave it up to you to discover its internals.
For any questions, just drop me an email.

Happy coding! :)

domingo, 26 de outubro de 2014

Kettle Database Value Lookup ignore case hack

Technology credits:
Java & Oracle RDBMS
Spoon (a.k.a Kettle)



Today i want to share with you a hack on the Pentaho ETL engine: Kettle.  I was investigating this tool, exploring the capabilities of Kettle to see what it could do for me. I've came to the conclusion that this is a very powerful tool that can do a lot, but also has it's 'shortcomings' especially when it comes to performance when dealing with a large amounts of data (around 1Gb).

Problem:
As i was testing this tool, i was trying to compare an input field from a file to a database column on the Oracle DB. As sometimes happens input data can sometimes be RaNdOm CaSe. That means the data can match if we ignore the upper/lower casing.

Solution:
Easy! Do a case insensitive match of the input! (you might say...) As i've come to the conclusion, it's not that easy to do that using Kettle... I wandered the web in search of clean and straighforward solution but i didn't find any so here is my solution:

First of all, convert the input field from file to upper case. This is easy, you can just run a simple JavaScript step to the input field and you are done.
Then you have to compare the uppercased field with the column from the database also in uppercase mode. Not so easy when the Spoon UI only enables you to select the table and column name on the 'Database Value Lookup' step, unless... you hack it!!!

Here's a screenshot of the hack:
What's the secret?
I've found out that the field 'Lookup table' is added to an SQL statement after the FROM keyword but...it's not as simple as that. You have to trick Kettle: Make sure your query has double quotes in it (such as "ID"), that's how the hack works. Kettle ends up injecting that SQL, requesting a column in uppercase mode from the database :)

Happy Coding!

quinta-feira, 11 de setembro de 2014

ActiveMQ integration testing with Maven

Technology credits:
Apache Maven & Failsafe
Apache ActiveMQ



Disclosure

Before you start reading further: Keep in mind that this is a work in progress, and although i'm a "Systems&Informatics Engineer" i consider myself a novice programmer. If you have any question you don't see answered here, please leave your comment on the section below to help me improve this article.
Also, you should be aware that this post was a result of a lot of work. If you like it, please feel free to show your appreciation by offering me a cup of coffee through paypal and/or add value to this article by sharing your knowledge.

Objectives of this tutorial:

  • Share how to easily start/stop an ActiveMQ broker for integration test purposes using Maven.
Pre-requisites:
  • You have knowledge about the Maven POM and know how to change it. If you need an example base project checkout this.

Problem

If you are developing and need some middleware like ActiveMQ it's always nice to be able to fully automate the start/stop of this kind of software that will be running as a service on a production server. You don't have keep remembering (to others or to yourself) the pre-requisites about how to build a project. The second advantage is that you always start from a clean, predictable state. And the third is that because you can stop whatever software you needed during Integration Tests, you keep your workbench nice and tidy, releasing memory usage and CPU time, allowing you to produce code faster and more efficiently.

Solution - ActiveMQ

If you are using ActiveMQ, the solution is easy, just add the following code in the 'plugins' area of your Maven POM and you are set:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
<plugin>
 <groupId>org.apache.activemq.tooling</groupId>
 <artifactId>activemq-maven-plugin</artifactId>
 <version>5.10.0</version>
  
 <executions>
  <execution>
   <id>start-activemq</id>
   <phase>pre-integration-test</phase>
   <goals>
     <goal>run</goal>
   </goals>
   <configuration>
    <fork>true</fork>
   </configuration>
  </execution>
  <execution>
   <id>stop-activemq</id>
   <phase>post-integration-test</phase>
   <goals>
    <goal>stop</goal>
   </goals>
  </execution>
 </executions>
</plugin>

Now just fire 'mvn clean verify' sit back and enjoy the magic ;)

Happy Coding!