Pro JSP gets 10 Horseshoes!
Gregg Bolinger has reviewed Pro JSP and gives it 10 horseshoes! Very nice - thanks Gregg! The book from which AppFuse was born...
The Angular Mini-Book is a
guide to getting started with Angular. You'll learn how to develop a bare-bones application, test it, and
deploy it. Then you'll move on to adding Bootstrap, Angular Material, continuous integration, and authentication.
Spring Boot is a popular framework for building REST APIs. You'll learn how to integrate Angular with Spring Boot and use security best practices like HTTPS and a content security policy.
For book updates, follow @angular_book on Twitter.
The JHipster Mini-Book is a
guide to getting started with hip technologies today: Angular, Bootstrap, and Spring
Boot. All of these frameworks are wrapped up in an easy-to-use project called JHipster.
This book shows you how to build an app with JHipster, and guides you through the plethora of tools, techniques and options you can use. Furthermore, it explains the UI and API building blocks so you understand the underpinnings of your great application.
For book updates, follow @jhipster-book on Twitter.
Gregg Bolinger has reviewed Pro JSP and gives it 10 horseshoes! Very nice - thanks Gregg! The book from which AppFuse was born...
This is a continuing series on what I'm doing to make AppFuse a better application in Winter/Spring 2004. Previous titles include: Changing the Directory Structure, Spring Integration
and Remember Me refactorings
.
- - - -
On my last project, we ported an existing JSP/Servlet/JDBC app to use JSP/Struts/iBATIS. In the process, I got to learn a lot about iBATIS and grew to love the framework (although I prefer to spell it iBatis). It was super easy to port the existing JDBC-based application because all of the SQL was already written (in PreparedStatements). Don't get me wrong, I think Hibernate
is the better O/R Framework of the two, but iBATIS works great for existing databases. The best part is that iBATIS is just as easy to code as Hibernate is. For example, here's how to retrieve an object with Spring/Hibernate:
List users =
|
And with Spring/iBATIS, it requires a similar amount of Java code:
List users = getSqlMapTemplate().executeQueryForList("getUser", user);
|
The main difference between the two is that iBATIS uses SQL and Hibernate uses a mapping file. Here's the "getUser" mapped statement for iBATIS:
<mapped-statement name="getUser" result-class="org.appfuse.model.User"> SELECT * FROM app_user WHERE username=#username#; </mapped-statement>
Spring makes it super easy to configure your DAOs to use either Hibernate or iBATIS. For Hibernate DAOs, you
can simply extend HibernateDaoSupport and for iBATIS DAOs you can extend SqlMapDaoSupport
.
Now to the point of this post: How I replaced Hibernate with iBATIS. The first thing I had to do was write the XML/SQL mapping files for iBATIS. This was actually the hardest part - once I got the SQL statements right, everything worked. One major difference between iBATIS and Hibernate was I had to manually fetch children and manually create primary keys. For primary key generation, I took a very simple approach: doing a max(id) on the table's id and then adding 1. I suppose I could also use the RandomGUID generator - but I prefer Longs for primary keys. Hibernate is pretty slick because it allows easy mapping to children and built-in generation of primary keys. The ability to generate the mapping file with XDoclet is also a huge plus.
As far as integrating iBATIS into AppFuse, I created an installer in contrib/ibatis. If you navigate to this directory (from the command line), you can execute any of the following targets with Ant. It might not be the most robust installer (it'll create duplicates if run twice), but it seems to work good enough.
install: installs iBatis into AppFuse uninstall: uninstalls iBatis from AppFuse uninstall-hibernate: uninstalls Hibernate from AppFuse help: Print this help text.
All of these targets simply parse lib.properties, build.xml and properties.xml to add/delete iBATIS stuff or delete Hibernate stuff. They also install/remove JARs and source .java and .sql files. If you're going to run this installer, I recommend running "ant install uninstall-hibernate". Of course, you can also simply "install" it and then change the dao.type in properties.xml. This will allow you to use both Hibernate and iBATIS DAOs side-by-side. To use both Hibernate and iBATIS in an application, you could create an applicationContext-hibatis.xml file in src/dao/org/appfuse/persistence and change the dao.type to be hibatis (like that nickname ;-). In this file, you'd have to then define your transactionManager and sqlMap/sessionFactory. I tested this and it works pretty slick. Click here to see my applicationContext-hibatis.xml file.
Some things I noticed in the process of developing this:
Hibernate is still the right decision for me, but it's cool that iBATIS is an option. Even cooler is the fact that you can mix and match Hibernate and iBATIS DAOs.
Tonight's Denver JUG should be good. The basic concepts meeting is on Hibernate and the main show is on Jabber: XMPP and Jabber Streaming Objects library. Nice! It's snowing like the dickens right now so getting there might be tough. Good thing I live close to downtown! The meeting is actually located in the same building where Julie works.
Update: Tonight's meeting was pretty good, but kinda boring. I didn't learn anything new in the
Hibernate session - but I did see Chris do a lot of stuff via command line (vs. Ant). I do everything in Ant, so
I was again able to see the beauty of Ant and Hibernate's <schemaexport> task.
The Jabber talk was interesting but dry - probably because Peter and Matt didn't
have a presentation or agenda for the meeting. Rather, they just stood up and talked about Jabber and its
XMPP Protocol. From what
they said, Jabber's streaming XML protocol is being used for a lot of things besides Instant Messaging. It's biggest feature seems
to be presence - the ability to know when someone (or something) is online. For using Jabber in your
Java applications, you might want to look at Matt's JSO Project. While
I'm at it, I might as well mention my article describing how to setup a Jabber Server.
I downloaded Mozilla Firefox (formerly known as Mozilla Firebird) for my Mac this morning and I'm definitely not impressed. I can't change the fonts and there's no scrollbar when I visit any sites. What a CF. As for Windows, I can't even download the installer. Too bad - this release had real potential.
I took some time last weekend and refactored AppFuse to use Spring to replace my Factories and Hibernate configuration. It only took me a couple of hours, which says a lot for Spring. I was amazed at how many things just worked. It actually lifted me out of my flu symptoms and made me feel euphoric. Or it could have been the Sudafed. In reality, I only replaced one Factory class (DAOFactory) - a fairly large class that instantiated DAOs using reflection and constructor variable inspection. I was also able to get rid of the ServiceLocator class, the getConnnection() stuff in ActionFilter and the hibernate.cfg.xml file.
The one thing I found when looking at the Petclinic and JPetstore apps was that they used an applicationContext.xml file for unit tests, and a (very similar) one for running the app in a container. To me, this was a warning sign. DRY (Don't Repeat Yourself) is a big reason for using XDoclet and I'm beginning to think that Spring could benefit from a little XDoclet lovin'. Anyway, back to the story.
I wanted to find a way to use the same XML files for testing and in-container execution. As you might know from Part I, AppFuse has 3 different tiers: dao, service and web. To run unit tests for the dao and service layers, I simply load a applicationContext.xml file in my JUnit test's setUp() method and go from there. I saw this in the petclinic app and found that it works pretty well. In the end, I decided to setup different XML files for each layer - applicationContext-hibernate.xml, applicationContext-service.xml and applicationContext.xml for the web layer. The main applicationContext.xml uses entity includes to reference the other two files.
The main pain I found was that the entity includes required different paths for tests vs. running in container. Basically, for tests, I had to use:
<!ENTITY database SYSTEM "applicationContext-database.xml">
While tests, using the ClassPathXmlApplicationContext required:
<!ENTITY database SYSTEM "WEB-INF/applicationContext-database.xml">
Using Ant to do a little replace logic allowed me to jump over this hurdle.
Using this setup, any new DAO definitions are added in src/dao/org/appfuse/persistence/hibernate/applicationContext-hibernate.xml, new Manager definitions (and declarative transaction settings) are be added in /src/service/org/appfuse/service/applicationContext-service.xml. The test-specific applicationContext-database.xml sits in the "test" directory and contains the following:
<bean id="propertyConfigurer" class="org.springframework.beans.factory.config.PropertyPlaceholderConfigurer"> <property name="location"><value>database.properties</value></property> </bean> <bean id="dataSource" class="org.springframework.jdbc.datasource.DriverManagerDataSource"> <property name="driverClassName"> <value>${hibernate.connection.driver_class}</value> </property> <property name="url"> <value>${hibernate.connection.url}</value> </property> <property name="username"> <value>${hibernate.connection.username}</value> </property> <property name="password"> <value>${hibernate.connection.password}</value> </property> </bean>
While the applicationContext-database.xml for the web is simply:
<bean id="dataSource" class="org.springframework.jndi.JndiObjectFactoryBean"> <property name="jndiName"><value>jdbc/appfuse</value></property> </bean>
To integrate Spring with my web layer (Struts), I just used the ContextLoaderListener in my web.xml file. I didn't see any point in bringing yet another JAR file into the mix.
<listener> <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class> </listener>
Finally, to expose Spring's context to my Struts Actions, I added the following to my BaseAction.java class:
private WebApplicationContext ctx = null;
|
This way, the UserManager implementation can be easier retrieved using:
UserManager userMgr = (UserManager) getBean("userManager");
|
The best part about the Spring integration in AppFuse is: (IMO) its Hibernate support and how it drastically simplifies my Hibernate DAOs (as if Hibernate wasn't simple enough already). I dig the ability to specify declarative transactions, and this refactoring seems to have reduced the "src" distribution of AppFuse by 2 MB (to 10MB total)! I don't know where this came from since the Spring JAR is almost 1 MB. The appfuse.war is about 500 KB larger, but I can live with that.
Of course, all of this has been checked into CVS if you'd like to take a look.
I tried posting the following to the tomcat-user list, but after not seeing it show up for 2 days, I'll just try it here.
----
I have developed a webapp that uses basic open source stuff (struts, hibernate, etc.). I've been developing/testing it on the following platforms - where it works fine and runs lickedy-split: OS X 10.3, 1.33 GHz, 1 GB RAM Windows XP, 2.6 GHz, 1.5 GB RAM Windows 2000 Server, 1 GHz, 512 MB RAM Windows 2000 Server, 1.5 GHz, 256 MB RAM All of these are running Tomcat 4.1.29 or 4.1.12. I recently tried to deploy it to the "test web server" (Windows 2000 Server) at my client and it runs dreadfully slow. It takes almost 30 seconds to load a page. The main difference is that the test web server has a 667 MHz CPU. It has 512 MB RAM, and we upgraded it to 1 GB, but that didn't help at all. I've recommended we get a faster CPU for the test web server, but I wanted to confirm that this could be the source of the problem. On this server, and the 1.5 GHz/256 MB RAM machine, IIS is integrated with Tomcat. All machines are running JDK 1.4.0 or above.
Any help is appreciated.
Folks that rag on Struts seem to point to ActionForms as one of its major design flaws. I've been slightly frustrated with ActionForms this week, but overall, I think they're a good thing. It's possible my ActionForm affection is misguided. The major reasons I like them is because I believe they allow me to do stuff that is not possible in other web frameworks. I definitely could be wrong though, so I'm hoping the other framework authors/users will speak up and say "My framework does that!" Specifically, I'm talking to the WebWork, Tapestry, JSF and Spring folks.
I do wish that I could throw my POJOs up to my UI, so I hope the following things are possible with the WTJS frameworks. It would simplify things if I didn't need to transform POJOs -> ActionForms (particularly with Hibernate).
I guess there's only two reasons I like ActionForms - the major one being the ability to specify (and generate) my client and server-side validation in XML. If I don't find this same slick feature in the other frameworks, I might have to do a bit of hacking to do the Interceptor with Validator thing - but hopefully I won't need to go there.
After seeing that Tim Bray is making $500/month of Google Ads, I decided it was high time I try to get this site approved again. Low and behold it worked! So they've approved me, but I only want to show them when folks come from Google. As in, when the referer (yeah, I know that's spelled wrong, but it is in Java and JavaScript too) contains "google" - show the ads. Anyone know how to do this cleanly in Roller/Velocity? Or JavaScript? I tried the following, but it doesn't work:
if (document.referer != null && document.referer.toString().indexOf("google") != -1) { // define variables document.write("<scr" + "ipt type='text/javascript' src='ads.js'><\/scr" + "ipt>"); }
Since Google is my top referrer - I think I'll get a fair amount of users seeing the ads, and it won't disturb the folks who come here just to read my ramblings.
Most of you won't care about this, but I need to write it down so I know the day that Daisy died.
Today was one of the first days we had a to make a real sacrifice as parents. Daisy was a cat that Julie had when I first met her. When Julie got Daisy, she was so small she could fit in the palm of her hand. Daisy was weaned too early from her mother (who was killed by a snake), so she had a personality disorder. Basically, she was mean and hated other cats. Daisy hated all the guys that Julie had ever dated (in college), until she met me. Julie knew it was special when she saw Daisy didn't hate me. Daisy traveled with us from Florida to Colorado when Julie moved here. She lived with us in our one-bedroom apartment, moved with us to our first house, and was a great cat for Abbie's first year.
Even though Daisy was mean, she was also the nicest cat I'd ever met. She'd always cuddle up on my lap when I watched TV or when I was working on the computer. You just had to watch out when you walked - she was an ankle biter. For the last year, she's been a pretty good cat with Abbie - just sitting there when Abbie poked and prodded. However, in the last month, she's started to fight back and has bitten and scratched Abbie quite a few times. My take on it was "good - Abbie should learn not to mess with the cat." Besides, Abbie wasn't crying - so it couldn't hurt too bad. Julie had a different take on it - especially since the folks at day care asked where the scratches came from.
When I got home tonight, Julie was in bed with Abbie at 7:00. "7:00!?" I thought, "that's a bit early." Then Julie told me "today was a very bad day" and she proceeded to tell me that they took Daisy in to the pound. Damn - my favorite cat. Ever. Gone. They got a new one, but it won't be ready until tomorrow. I want Daisy back, she was the bomb.
I love Panther's Exposé application. It's by far my favorite OS X application - I even find myself longing for it when using Windows. Could the wait be over? I stumbled upon WinPLOSION today, which looks (to me) to be Exposé for Windows. Anyone tried this thing. At $10, it's probably worth finding out if it works - but I'd thought I'd ask y'all first.