The easiest way to create a maintenance page for a .Net application is creating a static html page called
app_offline.htm and saving it to the root directory.
When IIS detects the existence of app_offline.htm, it will not only display the maintenance page, but also
return a “503 Service Temporarily Unavailable” error code, which is important to prevent search engine crawlers
from indexing the maintenance message as your site’s content.
A nice little trick: You could use data uri to embed images in your app_offline.htm, such as
1234567891011121314151617181920
<!DOCTYPE html><html><head><metacharset="UTF-8"><title>Maintenance</title><style type="text/css">#logo{width:216px;height:105px;background-repeat:no-repeat;background-image:url();}</style></head><body><divid="logo"></div><divlang="en">Here goes the maintenance message.</div></body></html>
A web site I’m working on uses Kentico CMS version 7. Occasionally its search function would break down due to a corrupted Lucene index.
Kentico CMS’s SmartSearch is built on top of Lucene.Net. Kentico 7 comes with Lucene.Net 2.1.0.3, a very old version. One of the things I tried was to upgrade Lucene.Net.
Although the latest version is 3.0.3, I decided to go with 2.9.4.1.
nuget install Lucene.Net -Version 2.9.4.1
Then I replaced Kentico’s Lucene.Net.dll and dependency ICSharpCode.SharpZipLib.dll.
As expected, Kentico throws an error “Could not load file or assembly ‘Lucene.Net, Version=2.1.0.3, Culture=neutral, PublicKeyToken=d846ee227f01c7a1’ or one of its dependencies. The located assembly’s manifest definition does not match the assembly reference.”
Trying to work around it, I added the following to web.config:
If your IIS 7 server sits behind a load balancer or a proxy server, it will always
record the IP of the proxy or load balancer as the client IP, which makes it impossible
to determine the origin of your visitors.
Recently I discovered Gatling, a load testing framework written in Scala, and used it to test a Kentico 7 web site.
I love it so much that I’ll ditch Apache JMeter from now on.
While both have a GUI to record tests, the major difference between them is JMeter tests are saved as .jmx files (XML),
but Gatling tests are .scala scripts written in Scala-based DSL.
Another interesting option would be Ruby JMeter, but I haven’t had a chance to try it yet.
This blog from flood.io points out that Gatling outperformances JMeter with over 20,000 concurrent users.
I must admit, however, that Gatling stil has a few rough edges, at least on Windows.
For example, some of my tests couldn’t finish because a report file’s file name was too long for Windows (260 characters maximum).
As a Ubuntu user and a Ruby enthusiast who prefers simple solutions, they’re a perfect combo.
I was just about to blog on how to set them up, when I found out this guy’s post beat me by a year. :-)
It’s a great post with step by step instructions.
I have just the following to add:
To set up your SSH key authentication, you can locate you Cloud 9 SSH key by going to “Your Account”, and then “Show SSH Key” under “Account Settings” on the right hand side.
Copy that key, click the Settings icon on the top right corner of your GitHub page, and then “SSH Keys” on the left side. Click “Add SSH Key” on the right to add your Cloud 9 SSH Key.
In case you use non-ASCII characters, for example, if you’re writing in German, you may get the followoing error when you run rake generate:
YAML Exception reading 2014-06-11-learn-german.markdown: invalid byte sequence in US-ASCII
/home/ubuntu/workspace/octopress/plugins/backtick_code_block.rb:13:in `gsub': invalid byte sequence in US-ASCII (ArgumentError)
from /home/ubuntu/workspace/octopress/plugins/backtick_code_block.rb:13:in `render_code_block'
from /home/ubuntu/workspace/octopress/plugins/octopress_filters.rb:12:in `pre_filter'
from /home/ubuntu/workspace/octopress/plugins/octopress_filters.rb:28:in `pre_render'
from /home/ubuntu/workspace/octopress/plugins/post_filters.rb:112:in `block in pre_render'
from /home/ubuntu/workspace/octopress/plugins/post_filters.rb:111:in `each'
from /home/ubuntu/workspace/octopress/plugins/post_filters.rb:111:in `pre_render'
from /home/ubuntu/workspace/octopress/plugins/post_filters.rb:166:in `do_layout'
from /usr/local/rvm/gems/ruby-2.1.1@rails4/gems/jekyll-0.12.0/lib/jekyll/post.rb:195:in `render'
from /usr/local/rvm/gems/ruby-2.1.1@rails4/gems/jekyll-0.12.0/lib/jekyll/site.rb:200:in `block in render'
from /usr/local/rvm/gems/ruby-2.1.1@rails4/gems/jekyll-0.12.0/lib/jekyll/site.rb:199:in `each'
from /usr/local/rvm/gems/ruby-2.1.1@rails4/gems/jekyll-0.12.0/lib/jekyll/site.rb:199:in `render'
from /usr/local/rvm/gems/ruby-2.1.1@rails4/gems/jekyll-0.12.0/lib/jekyll/site.rb:41:in `process'
from /usr/local/rvm/gems/ruby-2.1.1@rails4/gems/jekyll-0.12.0/bin/jekyll:264:in `<top (required)>'
from /usr/local/rvm/gems/ruby-2.1.1@rails4/bin/jekyll:23:in `load'
from /usr/local/rvm/gems/ruby-2.1.1@rails4/bin/jekyll:23:in `<main>'
from /usr/local/rvm/gems/ruby-2.1.1@rails4/bin/ruby_executable_hooks:15:in `eval'
from /usr/local/rvm/gems/ruby-2.1.1@rails4/bin/ruby_executable_hooks:15:in `<main>'
The solution is run export LC_ALL=C.UTF-8 first. To save time, I would join the commands as export LC_ALL=C.UTF-8 && rake generate.
(Some posts on the web suggest expoort LC_ALL=en_US.UTF-8, but that won’t work for the current version Cloud 9 at the time of writing.
To be sure, you can run locale -a to get a list of available locales.)
And then run jruby --command bundle install to get the gems ready.
Note: The tux gem works as a console to interact with my Sinatra application. I would also love to have the shotgun gem, but unfortuantely it’s unavaible for JRuby.
The JDBC driver (sqljdbc4.jar) for SQL Server is included in the source code, but if you prefer, you can also get it from Microsoft’s web site. We’ll just save it to our application root.
SQL Server table
Our application will use one table called “users”.
require'rubygems'require"sinatra"require"sinatra/activerecord"require'sinatra/form_helpers'require'active_record'require'active_record/connection_adapters/jdbc_adapter'require'sqljdbc4.jar'config_dev={:adapter=>"jdbc",:driver=>"com.microsoft.sqlserver.jdbc.SQLServerDriver",:url=>"jdbc:sqlserver://SERVER:1433;databaseName=DB_NAME",:username=>"USER",:password=>"PASS"}# JBoss connectoin pool. Need to set up datasource in JBossconfig_jboss={:adapter=>"jdbc",:driver=>"com.microsoft.sqlserver.jdbc.SQLServerDriver",:jndi=>"DATASOURCE_NAME"}# On JBoss $servlet_context is definedconfig=defined?($servlet_context)?config_jboss:config_devActiveRecord::Base.establish_connection(config)classUser<ActiveRecord::Basevalidates:consent,acceptance:true,presence:truevalidates:firstname,presence:truevalidates:lastname,presence:truevalidates:email,presence:true,format:{with:/\A[^@]+@([^@\.]+\.)+[^@\.]+\z/}endhelpersdodefh(text)Rack::Utils.escape_html(text)endendget"/users"do@users=User.take(10)erb:"users/index"endget"/users/:id"do@user=User.find(params[:id])erb:"users/show"endput"/users/:id"do@user=User.find(params[:id])if@user.update_attributes(params[:user])redirect'/users'elseerb:"users/show"endend
Our layouts and templates sit in the views folder with the following content.
Let’s get started by installing Warbler as a gem jruby --command gem install warbler. Next, we need to configure Warbler:
Create a new folder called config under our application root
Run jruby --command warble config, which generates a configuration template called warble.rb under the config folder
Edit config/warble.rb and update these lines:
config/warble.rb
12345678
# Application directories to be included in the webapp.config.dirs=%w(views)# Additional files/directories to include, above those in config.dirsconfig.includes=FileList['app.rb','sqljdbc4.jar']# Uncomment this if you don't want to package rails gem.config.gems-=["rails"]
Now we’re ready to package our app: jruby --command warble
A war file will be created as myapp.war.
Deploy the WAR file
I tried the JBoss 6 Admin console first, but got an Out of Memory error. I worked around it by copying the WAR file to the deploy folder (in my case
/usr/jboss6/server/default/deploy/) and restarting JBoss. And voila, our application is live on JBoss at http://jboss_address/myapp .