Thursday, 8 June 2017

Using RestTemplate to handle Http to Https Redirect

What is a RestTemplate? 

For the beginners of spring who don't know what a  RestTemplate api is for , this api comes as part of the spring-mvc module and it provides you client utilities to enable  HTTP/HTTPS communication to servers using the REST protocol. The methods provided by RestTemplate closely follows the REST principal.

 Problem Statement :

Suppose we are trying to perform a GET request to a specific URL and the URL returns a redirect response with a location header. 

For example :

GET /index.html HTTP/1.1
Server responds like this : 
HTTP/1.1 302 Found
Location: http://www.assortedminds.com/samples

How do you handle this using RestTemplate?
By default, RestTemplate redirects GET requests automatically. The default SimpleClientHttpRequestFactory of RestTemplate takes care of this redirect if the request is of GET type. But if the request internally redirects from one protocol to another, lets say from HTTP to HTTPS, the redirect doesn't happen by default. On a more serious note, this is not a feature of RestTemplate. RestTemplate internally uses the native JAVA Apis for handling http connections, so this behaviour derives from HttpURLConnection apis.
So We need to handle the redirect code manually in such cases.

The below code snippet is a sample of how we can handle such redirects.
 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
public String fetchSomeDataFromURL(String url) {

        //GET request to perform Rest Call
        ResponseEntity<String> response = perFormRestCall (url);

        //handle Redirects since HttpUrlConnection by design wont automatically redirect from Htpp to Https
        HttpHeaders httpHeaders = response.getHeaders();
        HttpStatus statusCode = response.getStatusCode();
        if (statusCode.equals(HttpStatus.MOVED_PERMANENTLY) || statusCode.equals(HttpStatus.FOUND) || statusCode.equals(HttpStatus.SEE_OTHER)) {
            if (httpHeaders.getLocation() != null) {
                response = perFormRestCall (httpHeaders.getLocation().toString());
            } else {
                throw new RuntimeException();
            }
        }

        return response.getBody();

    }

private ResponseEntity<String> perFormRestCall(String url) {
        try {
            return restTemplate.getForEntity(url.trim(), String.class);
        } catch (Exception ex) {
            throw new RuntimeException();
        }
    }
 

Notes: 

For 301(Moved permanently), 302 (Found) or 303(See other) responses, there should be a mandatory location header.
More details can be found in this link:

 https://www.w3.org/Protocols/rfc2616/rfc2616-sec10.html

Monday, 18 July 2016

[Eclipse Debugging] : Enable Remote debugging in wildfly JBOSS

Just try running standalone.sh script in debug mode :

./standalone.sh --debug 

This starts the jboss in debug mode, and the port 8787 is open for debugging. 

Start remote debugging from eclipse using 8787 as the debug port.

In case  your jboss is started as a service , locate your startup script  in /etc/init.d directory and
add --debug flag to the standalone.sh script invocation. This will help your service to always start in debug mode.


Monday, 21 March 2016

Reentrant locks vs Synchronized methods

Almost every java developer knows about synchronizing methods or code blocks using synchronized keyword, but very few knows abut synchronizing using Reentrant locks.

Before moving on to code samples directly, I would strongly suggest you to read the java doc for Reentrant lock :

https://docs.oracle.com/javase/7/docs/api/java/util/concurrent/locks/ReentrantLock.html

You will be already impressed by the number of utility methods provided by the lock, and how it helps in making a multi threaded application development easier.

Now lets move on to one of the simplest use case of using re-entrant lock , ie, implementing synchronized methods using re entrant locks.

This is how a synchronized method looks like :


 private void synchronized setValue(int myValue) {
        //setting a value for concurrent hashset
        value = value + myValue;
 }

The same can be achieved using a re-entrantlock as follows :

//define a reentrant lock as follows : 
final Lock valueLock = new ReentrantLock();

private void setValue(int myValue) {
        try {
            if (valueLock .tryLock(10, TimeUnit.SECONDS)) {
               value = value + myValue;  
            }
        } catch (InterruptedException e) {
            log.error("Exception while acquiring the lock!! ");
        } finally {
            valueLock .unlock();
        }
    }


Benefits of this approach :

We as developers receive more flexibility with locks . We can determine the time a thread has to wait for locks, we can even design a lock that follows fairness policy, and thus be impartial towards any thread. Developers get more control over the lock, which is an added advantage
 
 

Friday, 11 December 2015

Introduction to Gradle Jetty plugin


Introduction : 


Gradle comes with an inbuilt jetty plugin which can be used to deploy your web applications without any hassles. Your web application is build and deployed with a single gradle command, and this indeed makes a developer's  life less complicated.

Usage : 

Add the following to the build,gradle file.

apply plugin: 'jetty'  

Now, you can deploy your application in gradle using :

c:\assortedmind\gradle>  gradle jettyRunWar  
OR

c:\assortedmind\gradleProject> gradle jettyRun


As the name suggests, jerryRunWar will ideally deploy your war into the jetty server while jettyRun task uses the compiled files to be deployed into the server.

You don't need to give a build command before giving a "gradle jettyRun" command. This command takes care of compiling , building and deploying your web application into an embedded jetty server.

The URL is displayed at the command prompt window . The default http port used will be 8080.

Tips And tricks : 


Changing the default http port : 

Add the following lines to build.gradle file :

jettyRun {
httpPort = 9000
}


Debugging port : 

Wondering how to start jettyRun  task in debugging mode?  You need to set the GRADLE_OPTS variable  like this (in windows ):

C:\assortedmind\gradleProject>set GRADLE_OPTS=-Xdebug -Xrunjdwp:transport=dt_socket,address=9999,server=y,suspend=n

Now enable remote debugging in your IDE  by providing port number as 9999.
Now you are ready to go.

Need an upgraded version of jetty ? 

I was pretty happy using my jetty plugin tasks until I was  hit by a brutal  reality !! Gradle ships in an old version of jetty (version 6) which doesnt support servlet 3.0 and above. 
I would recommend using open source plugins for better support in terms of using latest versions of jetty or even tomcat server if needed. 
Plugins like Gretty is pretty impressive. I have tried it out and it works awsome! 


Referrences :

Gradle basics


Saturday, 11 July 2015

Spring : Configuring web.xml programatically

Spring 3.1introdced the support for Servlet 3's code-based configuration, which can be used as an alternative to the web deployment descriptor file web.xml. For this purpose, Spring has come up with 
an interface namely WebApplicationInitializer. 


The traditional web.xml file I have with me is as follows : 



And my servlet configuration file assortedmind-servlet.xml is as follows :



The servlet configuration is kept simple for the time sake. 

With these traditional xmls in mind, let us try to achieve the same purpose through a code based approach. The following piece of code can be used to replace web.xml



And the assortedmind-servlet.xml can be coded as follows :



Sunday, 7 June 2015

Gradle basics : Building war file using gradle

Gradle is very similar to the commonly used build management/dependency systems like ant scripts, pom.xml or apache ivy. But what attracted me towards gradle is its simplicity and availability of inbuild plugins which definitely makes our life easier. Also Gradle is readable , unlike its xml variants. 



Installing Gradle is pretty easy : 

Step 1: Download Gradle from the link : download link

Step 2: Unzip gradle to some folder of your choice .

Step 3: You can see the bin folder inside the unzipped file contents. You need to add this folder to the classpath so that you can run gradle command from any where on your system 

Step 4: now try running gradle from the command prompt. A similar output is shown: 


Writing your Gradle Build Script :

Gradle uses a Domain Specific Language (DSL) based on Groovy to declare builds. Groovy syntax is very similar to Java syntax, so it increases the readability of our build scripts. 
The Build script is a file named build.gradle.

The sample build.gradle file for my spring mvc application looks like this. Place the gradle file into the root of your web project folder.



apply plugin: 'java'
apply plugin: 'eclipse'
apply plugin: 'war'

repositories {
    mavenLocal()
    mavenCentral()
}

war {
    baseName = 'assortedmind-gradle'
    version =  '0.1.0'
}

dependencies {
    compile 'org.springframework:spring-webmvc:4.1.1.RELEASE'
    compile 'org.springframework:spring-context:4.1.1.RELEASE'
    compile 'org.springframework:spring-aop:4.1.1.RELEASE'
    compile 'org.springframework:spring-beans:4.1.1.RELEASE'
    compile 'org.springframework:spring-core:4.1.1.RELEASE'
    compile 'org.springframework:spring-expression:4.1.1.RELEASE'
    providedCompile 'javax.servlet:servlet-api:2.5'
  
}

task deploy(type: Copy) {
    from 'build/libs'
    into 'C:/Anushya/myJboss-jboss-as-7.1.1.Final/jboss-as-7.1.1.Final/standalone/deployments'
}

task wrapper(type: Wrapper) {
    gradleVersion = '2.3'
}


You might feel confused in the beginning, but later you will be overwhelmed by how simple this file really is...


A gradle script is a collection of tasks.  Gradle comes with inbuilt plugins , which are actually a collection of pre-defined tasks.  In our file,  the line 

apply plugin: 'java'
 is all it needs to add build related tasks. Type gradle task in the command line , and you can see the tasks that is been added as part of Java plugin. 

























Note :  To use the Java plugin for gradle effectively, make sure that you place your source files in src/main/java folder and non-java resources in src/main/resources. In this way , gradle will take care of packaging your war file by placing the class files and resources files in right manner. 

The Dependencies{ }  block lets you mention your compile time and runtime dependencies. 

The repositories{ } block lets you mention which repository needs to be referenced during tasks execution. First it will look for the jar files in local repository , and if not available , will check for maven central repository.

Type gradle build to build a war file. The war file will be present in the build/lib/ folder. 

My deploy task copies the war file into my jboss standalone deployment folder.

The complete web application project structure is as follows : 





















Finally the wrapper task lets you run this gradle script in machines that does not have gradle installed.  Run the task and it will install gradle for you. 



Sunday, 12 April 2015

Simple fix for mysql " Access denied for user 'root'@'localhost' " issue

There are numerous instances where this issue can be faced. In my case, I faced this while installing mysql in my RHEL machine . I installed rpm as root user, and after installation , I faced this issue while logging into mysql .
ERROR 1045(28000) : Access denied for user 'root@localhost' (using password: no )

Even
 mysql -u root -p ***
produced the same error.

This issue can be fixed in a very quick and easy solution .

If you have started mysql , stop it.
Now restart your server with  --skip-grant-tables option
eg : in my case  I used the command
sudo /etc/init.d/mysql start --skip-grant-tables

Now you dont need a username password combination to open mysql.
simply, use mysql to login to mysql prompt. 
 shell> mysql

now load the grant tables :

mysql> FLUSH PRIVILEGES;

mysql> SET PASSWORD FOR 'root'@'localhost' = PASSWORD('HelloWorld');

This command will help you set your new password.

Dont forget to restart your mysql server with the new user-name and password combination, as opening mysql with --skip-grant-tables is insecure.

hope this helps you guys out :) Good luck.