Monday, August 23, 2010

Logging inside a web application using log4j

It is always desirable to have a logging feature for the application, especially when it is interacting with a range of external systems. It would help in traceability and troubleshooting and localizing the problem when things go wrong.

The following is the steps you do for enable logging
1. Download log4j-1.2.16.jar from apache web site and put this in your WEB-INF/lib folder

2. In your web.xml ( preferable your first servlet that executes )
   
       
       

        Login
        Login
        Login
    init-param

        param-name
             log4j-properties-location       

       /param-name
      param-value
       WEB-INF/log4j.properties   
      /param-value  
   /init-param

3. Create a simple text file  log4j.properties and place under WEB-INF folder of web application

---
log4j.rootLogger=DEBUG, R
log4j.rootLogger=TRACE, R

log4j.appender.R=org.apache.log4j.RollingFileAppender
log4j.appender.R.File=${catalina.home}/logs/myapplication.log
log4j.appender.R.MaxFileSize=10MB
log4j.appender.R.MaxBackupIndex=10
log4j.appender.R.layout=org.apache.log4j.PatternLayout
log4j.appender.R.layout.ConversionPattern=%p %t %c - %m%n

---
4. a) Say in Login servlet init() method

        String log4jLocation = config.getInitParameter("log4j-properties-location");
        ServletContext sc = config.getServletContext();

        if (log4jLocation == null)
        {
            System.err.println("*** No log4j-properties-location init param, so initializing log4j with BasicConfigurator");
            BasicConfigurator.configure();
        } else
        {
            String webAppPath = sc.getRealPath("/");
            String log4jProp = webAppPath + log4jLocation;
            File yoMamaYesThisSaysYoMama = new File(log4jProp);
            if (yoMamaYesThisSaysYoMama.exists()) {
                System.out.println("Initializing log4j with: " + log4jProp);
                PropertyConfigurator.configure(log4jProp);
            } else {
                System.err.println("*** " + log4jProp + " file not found, so initializing log4j with BasicConfigurator");
                BasicConfigurator.configure();
            }
        }

        mlog.info("LOG4J initialization for CCM User application successful.");   
       
 ...
4. b )
In the servlet class ( in post method or what have you ) and POJO class deployed
At start declare

Logger mlog = Logger.getLogger(some.class);
..
..
// some action

mlog.info("Inside 'Add UC1 ' request processing.");
mlog.trace("Query executed " + ps);
mlog.trace(ackmsg);
mlog.error(ex.getMessage());

5. Run the application perform actions that would generate these logs.
6. Go to TOMCAT installation %HOME% and open myapplication.log in a text editor.

That it ; you have enhanced traceability of your functionailty you coded.

References :
http://www.avajava.com/tutorials/lessons/how-do-i-initialize-log4j-in-a-web-application.html

Tuesday, August 10, 2010

Typical web application developement activities

Objective :

The prime motives behind daily incremental backup of web project ( say for example : ccm user app ) Eclipse workspace during a web application development are

- To incrementally add functionality
- To keep the build always successful.
- Track the various technological solutions behind implementation.


I shall use the terminology Sprint[1] cycle against the date terminology to go in
chronological order.

SPRINT [0]
- Requirements Document is ready and communication with customer on way.


SPRINT [1]

- Get use to the JEMT ( Java/JEE , Eclipse IDE , MySQL DB, Tomcat Web/App Server ) Skills
- Setup and configuration of local development box.
- Come up with a Skeletal application.
- Primary page navigation working. ( i.e hook JSP with their servlets )


SPRINT [2]

- Fine tune login page, authentication against DB. ( Until AXL dependency is solved by another developer )
- Database schema seeding , Use of MySql Browser to have some frequent queries at your finger tips for testing.
- Java servlet classes for primary use case handling into the workspace.
- Helper beans in separate packages -  a) DB connection b)  Mailer Bean

SPRINT [3]

- Design document in parallel to be drafting for better elaboration.
- Sequence diagram drafted online
- JSTL technology ; View Bean iteration ; Combo box fetching DID details done.
- Reports using same JSTL technology.

SPRINT [4]

- Properties file for application creation. Reading it during login and populate in helper-Bean - MyAppProperties
- MailServlet alone tested and properties file reading also successful.
- DB operations for various prime use flows done. JDBC technology used.
- Tested application as whole till now.


SPRINT [5]

- Look and feel ; to be in left controls ; header and center region done. Frameset / DIV HTML technology.
- Integrate Javascript for some client side validations.
- Range DID inputs done. i.e bulk insertion of rows not violating the primary constraint.
- Migration solution done. POI technology for xls sheet + Self executing jar with in built help done.
- Testing and deploy  ( Bundling of dependent jars with web application ) scenario till now.
- Deploy guide steps in parallel.

SPRINT [6]

- Location code values to be read from properties file done. Important datastructure LinkedHashMap usage
- Generic resource messages JSP + Message bundle integrated.
- Various boundary cases simulated and tested.
- Helper beans in dedicated packages - for AXL system integration done.
- Additional reports use cases done and tested to aid the customer.
- Comprehensive error/message reporting.
SPRINT [7]

- Migration to CCM 6.1 ; AXL issues hit during the stuff. Good in one sense that we were able to catch in-house
  rather than at production site
- Pre-conditions and Post conditions check for AXL external system interaction done.
- Implemented logging using Log4J. Improves traceability and troubleshooting down the line.
- Testing against target platform CCM 5.1 done. ( Setup and licensing issues in-house to be dealt with )

SPRINT [8]
- Implemented pagination. Refined UI.
- Deployment guide drafted and done.
- Testplan created for basic sanity purpose.

SPRINT [9]
- Customer demo . Deployment on production
- Defect fixing mode and enter into maintenance mode.

A lot of time goes in googling for some basic information. If you can index these pages for reference, it would help in quick development and troubleshoot.

Reference links

JDBC Example with MySQL Working with sessions Display Data from Database in JSP The Ultimate JSP Tabs! E-Surfer's Weblog: Thinking,feeling in JAVA, J2EE, XML, Web Services development JSP - jstl-standard-1.1.2.jar - Jakarta Standard 1.1 Taglib - Standard Simple Web Application using Servlet How to authenticate in a webapplication using jsp and servlet | s2sgateway All Classes (POI API Documentation) on Apache Servlet and JSP development with Eclipse WTP HTML Forms tag lib JSP, on click , forward to new JSP - Google Search Server-side Java: Understanding JavaServer Pages Model 2 architecture - JavaWorld Servlet form processing example, Handling HTML form submission using Servlet, form servlet Step by step J2EE web development tutorials,Servlet tutorials How to use jsp page to display data from a database table - DevX.com Forums JSP Standard Tag Library (JSTL) c:forEach Tag Retrieve data from database by using sql tag of JSTL SQL library InformIT: Using HTML Forms with Servlets > The FORM Element Javascript Date Validation 'Re: Tomcat 4.1.24 :Cannot forward after response has been committed' - MARC JSTL SQL query taking parameters JavaScript Form Validation : quick and easy! Dynamically add button,textbox,input,radio elements in html form using JavaScript | ViralPatel.net  JSP Syntax  WTP Tomcat FAQ - Eclipsepedia An ordered HashMap in Java — Possibility and Probability Documentation - Cisco Developer Community All versions AXL APIs Client-side HTML table pagination with JavaScript - NewInstance JSP Controls Tag Library AJAX MODE 

Sample structure 
Folder PATH listing for volume System
Volume serial number is 0006EEA4 4E36:2815
C:.
|   .classpath
|   .project
|   project_structure.txt
|  
+---.settings
|       .jsdtscope
|       org.eclipse.jdt.core.prefs
|       org.eclipse.wst.common.component
|       org.eclipse.wst.common.project.facet.core.xml
|       org.eclipse.wst.jsdt.ui.superType.container
|       org.eclipse.wst.jsdt.ui.superType.name
|      
+---build
|   \---classes
|       |   AddDIDServlet.class
|       |   AddUserServlet.class
|       |   DeleteUserServlet.class
|       |   HomeServlet.class
|       |   Login.class
|       |   POIExcelReader.class
|       |   SearchUserServlet.class
|       |   ViewDIDServlet.class
|       |   ViewUserServlet.class
|       |  
|       \---ccmusers
|           \---util
|                   AXLBean$MyTrustManager.class
|                   AXLBean.class
|                   CCMMessageBundle.class
|                   DBConnection.class
|                   LocationBean.class
|                   MailerBean.class
|                   MyAppProperties.class
|                   TableData.class
|                   users_data.class
|                  
+---notes
|       code_snippets.txt
|       customer_input_feedback.txt
|       deploy_steps.txt
|       important_solutions.txt
|       meetings_and_discussions.txt
|       migration_notes.txt
|       resources.txt
|       sample_sql_queries_dev_n_test_status.txt
|      
+---src
|   |   AddDIDServlet.java
|   |   AddUserServlet.java
|   |   DeleteUserServlet.java
|   |   HomeServlet.java
|   |   Login.java
|   |   POIExcelReader.java
|   |   SearchUserServlet.java
|   |   ViewDIDServlet.java
|   |   ViewUserServlet.java
|   |  
|   \---ccmusers
|       \---util
|               AXLBean.java
|               CCMMessageBundle.java
|               DBConnection.java
|               LocationBean.java
|               MailerBean.java
|               MyAppProperties.java
|               TableData.java
|               users_data.java
|              
+---util_scripts
|       ccmusers_v1.sql
|       ccmusers_v2.sql
|      
\---WebContent
    |   adddidpage.jsp
    |   adduserpage.jsp
    |   content.jsp
    |   deleteuserpage.jsp
    |   footer.jsp
    |   gen_validatorv31.js
    |   header.jsp
    |   home.jsp
    |   invalid.jsp
    |   leftcontrols.jsp
    |   login.jsp
    |   message_n_error_display.jsp
    |   paging.js
    |   searched_results.jsp
    |   searched_users.jsp
    |   searchuserpage.jsp
    |   testpagination.jsp
    |   viewdid_info.jsp
    |   viewuser_info.jsp
    |  
    +---images
    |       cal.gif
    |       ccd_application_logo.gif
    |       header.gif
    |       home_icon.gif
    |       line.gif
    |      
    +---META-INF
    |       MANIFEST.MF
    |      
    \---WEB-INF
        |   ccmuserapp.properties
        |   web.xml
        |  
        +---lib
        |       activation.jar
        |       jstl.jar
        |       mail.jar
        |       mysql-connector-java-5.1.13-bin.jar
        |       standard.jar
        |      
        \---tld
                c.tld
                sql.tld
               
Hope this helps in getting the basic framework ready.

Thursday, August 5, 2010

Creating an executable jar with dependent jars.

The following is skill you would require for packaging of a deliverable

  1. Set Java path like
          C:\setjdk16.bat
   2. Copy this java class into temporary dir
          cp Test.java C:\tmp\
   3. Switch to tmp directory
          cd C:\tmp
   4. Copy dependent jars dep1.jar dep2.jar C:\tmp
   5. Compile Java class
          javac -cp ./dep1.jar;./dep2.jar Test.java
   6. Create manifest.mft text file with contents like

Manifest-Version: 1.0
Main-Class: Test
Class-path: dep1.jar dep2.jar


   7. Archive contents
          jar cvfm myexe.jar manifest.mft *.class *.jar
   8. Run output jar like
          java -jar myexe.jar


To run with arguments
java -D -jar myexe.jar args[0] ...

Many a times even simple tend to forget this helpful concept

Note as on 17 OCT 2013 when using 1.7 JDK manifest.mft should look like else classloading of dependent jars fails  ( attribute Created-By mandatory )

Manifest-Version: 1.0
Main-Class: SSHCommandExecutorv2
Class-path: jsch-0.1.44.jar
Created-By: 1.7.0_06 (Oracle Corporation)

Wednesday, August 4, 2010

Parameters for resource messages

Spent a lot of time thinking/recollecting to do this.
Here it goes



import java.util.ListResourceBundle;
/*
** Resource bundle test
*/
public class AppMessageBundle extends ListResourceBundle
{
 
    static final Object[][] m_contents =
    {
    // KEY                  , MESSAGE

    // Messages for the migration exception log
     {"HELLO"         , "Hello without parameter."
     }
    ,{"HELLO_ARG1"    , "Hello {0}."
     }
    ,{"HELLO_ARG2"    , "Hello {0} and {1}."
     }


    // Errors and Exceptions
   

    // Genric Messages
    // To add as and when requirements arise

    // Command line usage
    // To add if requirements arise.

   
  };


  public Object[][] getContents()
  {
    return (m_contents);
  }
}




import java.text.MessageFormat;
import java.util.*;

/*
** Test program to understand 
** passing arguments to construct
** resource messages of more meaning.
*/

public class AppMessageBundleTester
{
  private static final String RESOURCE_BUNDLE = "AppMessageBundle";
 
  public static void main ( String[] args )
  {
      String name1 = "Test string";
      int num = 734;
      printResourceMessage("HELLO");
      printResourceMessage("HELLO_ARG1",new Object[]{name1});
      printResourceMessage("HELLO_ARG2",new Object[]{name1,name1});
      printResourceMessage("HELLO_ARG2",new Object[]{name1,num});
     
     
  } 
   private static ResourceBundle s_resBund = ResourceBundle.getBundle(
                                              RESOURCE_BUNDLE, Locale.getDefault());
    public static ResourceBundle getResourceBundle()
  {
    return s_resBund;
  }

  
  static void printResourceMessage(String message, Object[] names)
  {
    println(MessageFormat.format(s_resBund.getString(message), names));
  }

  static void printResourceMessage(String message)
  {
    println(s_resBund.getString(message));
  }
    static void println(Object o)
  {
    System.out.println(o);
  }

   
   
} //  end of class
This is all java and you need not have to go through hassle of having properties file. Specially useful in a web application.

Followers