How to win the lottery…
… easy, pick the winning numbers!
The trouble is there are 45 million combinations!
So, why not share your numbers with everyone else to ensure we all use different numbers.
If enough people do it, then we’ll ensure someone wins!
Note – this is not a syndicate. The winnings will not be shared. The idea is just to ensure that someone wins!
No details will be recorded – so we can’t hassle the winners.
You either tell us the numbers you’re going to use (anonymously) or we give you numbers that no one else has selected (i.e. like a lucky dip).
Ok, it’s just an idea at the moment… so, who is in?
JRubyConf EU 2015
Introduction
I was fortunate to attend the recent JRuby conference in Potsdam (nr Berlin).
http://2015.jrubyconf.eu/
JRuby is an implementation of the dynamic language Ruby based on the JVM – this gives it the advantage of the ubiquity of the JVM, the more mature threading facilities and access to the various Java libraries/frameworks etc. Jruby is an interpreted language – by the JVM.
It can be used in many ways – from pure Ruby without touching Java or it can be embedded as a library within a Java app. You can both call Java from JRuby and vice versa.
The conference was a one day event, with 10 varied talks.
There was even quality coffee on tap (pic)
You can see the talks here – http://confreaks.tv/events/jrubyconf2015
Talks
Keynote – JRuby 9000 has recently been released, which includes a major revamp of the core engine and a rewrite of the IO/Transcoding parts to make them more native – thus making them more compatible with the C implementation of Ruby.This was presented by the amazing Charles Nutter and Thomas Enebo.
JRuby/Gradle build tool – this talk was about how Gradle could be used to provide a better build tool for mixed Java/Ruby projects – covering both Java and Ruby dependancies. It can also be used to build a single jar file for deployment.
JRuby/Java data processing – based on the Java library Esper, a Ruby specific library has been built that allows for stream processing within Ruby.
A related tool is Embulk for bulk data loading.
This was followed by a talk about Shoes – which is a Ruby based UI library – its recently been brought uptodate and compatible with JRuby – its used to power the learning website – http://www.hackety.com/
The next talk was on Mirah- a JVM language that is similar to Ruby but is typed and compiled to bytecode, this gives it the advantage of better performance over JRuby. Its still in a very early stage – http://www.mirah.org/. The word “mirah” means “ruby” in Javanese, the language of the island of Java. Get it? “Ruby” in “Java”?
Alex Coles did a talk on the Quiet and Versatible Rubyist. There are times when we can work with the technologies we love. And there are times when we have to work with WebSphere and Oracle. Sometimes it’s necessary to be pragmatic.
This was followed by Joe Kutner’s talk on the 12 factor app – 12factor.net
– codebase/use version control, separate repo per app/service
– manage dependencies. e.g. jbundler
– configuration, per env settings eg DB password – don’t check it in!
– backing services, eg DB etc. Attachable resource, eg via env
– build, release, run
– processes.
– port binding. Do in app not via other service – containerless
– concurrency. scalability. Services.
– disposability. Start quick, resilience to failure. Shutdown quickly/ gracefully
– dev/prod parity, identical! Same tools/services
– logs
– admin processes. DB migrations etc.
Next we had Scaling JRuby with docker and Apache mesos. Michal Olah/ Rocky Jaiswal
– mesos. Treat cluster as single resource. W/zookeeper. Distributes tasks to nodes.
– docker containers run on the nodes, lightweight, isolation
– marathon executor service. Hosts docker containers.
– grouping servers, eg by location
JRuby file IO using mmap – Colin surprenant. This talk was about getting fast disk IO within JRuby and where to draw the line between Java and JRuby objects.
Diversity in IT
10% world is white male, yet 90% of IT is dominated by white males! Studies have shown that half the women who enter the technology field will leave over time.
We need to promote and encourage diversity in our industry.
There were then a few short lightning talks – from multi-dimensional analysis to a new systems programming language!
Conclusion
All talks were very interesting – providing me with several tips and tricks and potential future ideas.
Want a lightweight in memory database – try Boon!
So you have a million objects in memory, you’ve got them in a Map – so you can access them by key very quickly, but now you want to find all that match on another field. Well, they are in memory, so you can go through them all. However that is going to be slow.
[code]
static Collection
You could stick them in a DB (like MySQL or MongoDB) however it will still be pretty slow to query/convert to/from or you could write your own query engine from scratch. Admittedly its optimised for that, but it’s still a separate process and the data has been converted into your
If only you could index the collection by that other field…
Well you can – just put the collection into a Boon Repo and add an index on that field and it will take care of ensuring the index is kept up to date.
[code]
Repo<Integer, Map> dbRepo = Repos.builder()
.primaryKey(“name”)
.lookupIndex(“colour”)
.build(int.class, Map.class);
…
dbRepo.addAll(database);
…
List
[/code]
For these simple queries the search changes from O(n) to O(1) – bucket loads quicker!
It provides a rich set of criteria facilities that allows you to query in a variety of ways – equality, in, between, etc and these can be composed using and/or …
[code]
List
[/code]
It effectively provides an In memory object database, searchable, you define the indexes you want to use – doesn’t have to be unique.
– its pure Java (1.7) – although I have a 1.6 backport.
Ideal for the case where you have low frequency updates but lots of varied queries.
Its a little quirky when it comes to updating items – to update an item, you need to either delete/re-add it or get the existing object, clone it, amend it and re-add it.
[code]
managers = dbRepo.query(eq(“job”, “manager”));
Map updatedMgr = Maps.copy(managers.get(0)); // Maps.copy is a boon cloning util
updatedMgr.put(“colour”,”blue”);
dbRepo.modify(updatedMgr);
[/code]
I have found performance to vary somewhat on more complex queries – but I am sure that is something that will improve greatly.
As with all libraries, you need to use it and see if it helps with what you need.
Links to similar data repo articles
– Boon’s Data Repo by Example
– Boon Data Repo Indexed Collections and more
– Background on Data Repo
– What if Java Collections and Java Hierarchies were Easily Searchable?
– Unrelated, but… Boon beer
Thanks to Rick Hightower for giving us this boon!
@skillsmatter presents @unclebobmartin ‘s TDD and Refactoring course
A few weeks ago I attended Uncle Bob’s TDD and Refactoring course – here are my highlights…
Number one for me was this quote “TDD is double entry bookkeeping for programmers!”
Skills You Need When Looking at code:
– able to identify problems
– knowing whats better
– transform in small steps towards the better version
When the code makes you do something ugly – the design is wrong!
An interesting presentation style point was the intermission discussions – completely unrelated. A few of us were trying to work out how they related to the course, until someone asked and found out they didnt :)
Thanks again, Mr Martin.
WoW Activity Feed – the tech behind it…
As mentioned previously – its live!
The app is built on nodejs, using coffeescript with a MongoDB database.
I chose coffeescript, as I prefer the Ruby-like syntax and really don’t like Javascript curly-brace’s. I also found NodeJS to be very fast and lightweight. Also, the callback model in node takes some getting used to.
Libraries like async – help make the callbacks more manageable and avoid the pyramid of death.
I used the rapidly growing expressjs framework – which was great at keeping out of your way and letting you just do what you need.
HTML was put together via Jade templates and Stylus stylesheet helper.
The core of the app accesses the Armory via a great little library – node-armory.
The data from the armory is saved directly as JSON to MongoDB and as further updates come in, the jsondiffpatch library – is used to determine whats changed.
There are some tests written with mocha and sinon for stubs/mocks. My testing style is to use them when there is a problem – so they are probably broken at the moment.
I found a few features missing with the node-armory library and so forked the project to address them, such as support for using a proxy (for debug purposes) and using Mike Reinstein’s version of the http lib request that supports compressed request/replies.
See package.json for details of all the libraries used.
A couple of MongoDB’s features that came in handy, like:
- Time to live (TTL) collections – records automatically removed when a date field is older than specified. Useful to ensure only recent data is kept.
- Capped collections – limit how big a table gets, automatically throws away old records when it gets full. Good for logs.
- Aggregate queries – a map/reduce like facility for querying the DB.
Definitely the worst bit of the code is the RSS feed entry formatting – need to think of some ways to refactor it sensibly.
To make the search feature nice and responsive, backbone was used in a very basic way.
The main thing I missed from Rails was the asset pipeline, which lets you combine all the client side assets (javascript and css files) into just a few minified files.
One of the most interesting parts of the app (at least to me), is the scheduled job that checks the armory for updates. The core of which uses an async queue to kick off many calls to the WoW API and collect the results.
Rails 3.2.11 upgrade (lorra lorra security issues…)
There have been quite a few Rails issues over the last few weeks… so I better upgrade mine too :( Note I use rvm with gemsets to separate gem versions – probably could stop using gemsets given the latest bundler, but havent got on that bandwagon yet.
Bingo Caller!
link Not currently live on a public site, but good basic example to test the upgrade. This is currently rails 3.2.8 (rails 1.9.3/sqlite), which is not too far behind the latest, 3.2.11 – so it should be easy… First step, rvm implode and re-install – I have lots of rvm/gemset cruft and now seems like a good time to tidy that up. Then run bundle to get the gems that should currently work. Then realised that rvm wasnt loaded properly and so the gem installs did not go into my gemset – so re-started the terminal session and ran bundle again (probably could have source’d rvm, but probably better this way). Make sure the db is up to date (rake db:migrate). Then try running the app (rails server) [yup – no tests…]. Seems to be working fine – thought for a moment I was done, but then realised I have not upgraded rails yet – doh! bundle update Now I am on rails 3.2.11. Lets try again – rake db:migrate; rails server. And seems to be working ok. Checked the log files – one error about binary data in a string field (encrypted_password) – but then we had that previously. Also used the rails_admin gem for a quick built in db-viewer – thats working ok too. Time to see if it works on Heroku too…
- Install the Heroku Toolbelt
- (re-) create the app – heroku create –app bingo-caller
- push the code – git push heroku master
- Which failed :( – something about sqlite3 – but that shouldn’t be in prod mode…
- Gemfile did not specify ‘pg’ for Postgres – so amended it to say sqlite3 for dev/test and pg for prod (group :production do … end)
- This deployed, but then got an error when trying the app ( http://bingo-caller.herokuapp.com )
- Maybe it needs the db setup (heroku rake db:migrate) – that also failed “undefined method `database_authenticatable’ for #<ActiveRecord::ConnectionAdapters::PostgreSQLAdapter::TableDefinition:0x00000004a16f68>”. Looks like some Devise related issue.
- Mmm – lets try re-creating the sqlite db – does that work? Nope – same error.
- This was a Devise 2.1.2 to 2.2.1 upgrade… a quick google highlights that the schema layout has changed. Re-organised as per that link and now its deploying/working ok.
Humming Now
link Largely a javascript based site – but does use Devise, so that upgrade might make it problematic. Rails 3.2.1 (ruby 1.9.2/sqlite/postgres)
- Decided to make sure gcc was good (surprised I didnt get issues above…)
- Installed brew.
- Install brew gcc etc utils, see here and here.
- Install 1.9.2 (seemed to be more reliable for this app, ssl related issues – although that could be an issue with my mac config)
- Now we can get to the app, run bundle to install the current gem versions.
- but that failed to find one gem version – “Could not find jquery-rails-2.0.0 in any of the sources” – strange, has it been removed? (shows as ‘yanked’ here – http://rubygems.org/gems/jquery-rails/versions )
- So, trying to get the latest version, via ‘bundle update jquery-rails’
- But then found another missing one – “Could not find twitter-2.0.2 in any of the sources”
- So maybe I will just update them all :) ‘bundle update’ – phew, that seems to have worked.
- Moved aside the sqlite db and tried ‘rake db:migrate’, which got the devise/db auth error as above.
- So followed the above fix, re-did the schema and that worked.
- Tried rails console and that seems ok
- Then tried rails server, got a warning from Devise – “Devise.use_salt_as_remember_token is deprecated and has no effect. Please remove it”
- The home page seems to load, but there is a javascript error behind the scenes, looks like its not escaping the json stuff ” … Router({'url_root&#x …”
- Looks like ruby/erb is encoding the model json stuff – noticing that I have <%== (double equals) in the other js parameter, which is working – so I tried that in the first one and it worked. Cant seem to find it defined anywhere, but presumably its to not escape the value.
- Tried running rails console – but got [BUG] Segmentation fault. Then ran rake db:migrate and tried rails c again and it worked (also changed directory elsewhere and back…)
- The Twitter gem used has moved on from 2.0.2 which is currently used – latest is 4.4.4, in particular, Twitter has removed the public timeline, which this site uses.
- Tried to remove public timeline, but getting issues with current_user not being defined.
- Tweetstream gem allows access to public timeline via the streaming API – thats worked :) , but entities/link urls are coming through strangely… :(
- Getting some issues around the user/session. It seems to let me login (via Twitter) and the user home page comes up. But any subsequent requests are failing – it finds no current user. Its like the session is losing the user or its being re-created.
- I then reverted those changes and tried to just get the current app running and I am still getting the same issue. I had to update some gems which had been yanked, but then that led to other gems getting updated… But have fixed the Rails/devise/rack/omniauth gems at their original versions.
- I would think that devise/rails looks after the user in the session – so that would be the place to look – but now I have pinned those to the current live version, that doesnt seem likely…
- Time to try a different approach – just patching the current Rails version – see if that can be done on Heroku…
- As a couple of the gems have been ‘yanked’, I amended the Gemfile to use github versions, like so
-
gem ‘twitter’, :git => ‘https://github.com/sferik/
twitter.git‘, :tag => ‘v1.6.0’ - And it worked – well, now I have a running app, time to patch it!
- Best link I found is this Engine Yard one.
- Before/after I tried this exploit tester – but it seemed like it was ok before/after.
- From the above work trying the big bang update, I re-applied the changes to get the public timeline working via tweetstream. Oops – just noticed search does not work… something for later.
- And then pushed it to production, next!
Quoter
Largely a backend site, but has a few public urls with stats on. Use Mechanize and Savon, will probably need to update these too.
- Following the steps above, via rvm/bundler – get all the required gems in place.
- First, lets try the exploit tester… says its not vulnerable (at least without logging in)- but better safe than sorry…
- Ok, lets try a full update of Rails – currently on 3.2.3… change the Gemfile and run ‘bundle update rails’
- But that didnt work – seems some old gem is causing issues, getting error “NoMethodError: undefined method `field_changed?'”
- so lets try ‘big bang’ – ‘bundle update’ …
- It seemed to work, but needs more testing ….
- So, went for the patch option from the Engine Yard link above. Did some testing, seems ok.
- And then pushed to prod :)
Image Site
Not deployed, so perhaps leave… last version used – 3.0.9 …
Other useful links:
- http://railsapps.github.com/updating-rails.html
- https://groups.google.com/forum/?fromgroups=#!msg/rubyonrails-security/61bkgvnSGTQ/nehwjA8tQ8EJ
World of Warcraft Activity Feeds – now live!
Hi,
A few years ago, the WoW Armory had a way (unofficially) to access a feed of updates for your in game character, eg gaining a level etc. (or so I seem to remember, but maybe my mind is playing tricks…).
Then along came the new Armory site and that feed wasn’t available anymore :( ..
Last year, Blizzard came out with an API to access WoW character progress data and it was thought that an official feed would be produced. However 12 months later, there is no sign of a feed.
Recently I wanted such a feed, had a look around and finding none – decided to write one. And its now live http://wowactivity.kimptoc.net/ :)
As users of the WoW API have to be open source – the code is here on github.
The feed providers both character and guild RSS feed. Its largely based on the news/feed items that come with character and guild lookups but it also does track changes, eg when level changes. This RSS feed can be used in many ways, from piping it into a feed reader to sending it to twitter, facebook or guild websites.
Its currently tracking over 150 characters and guilds.
Its built using coffeescript (aka javascript) and MongoDB – will probably do a separate post on that.
Enjoy!
Chris
Ruby Timeout problem – the cheaters way out ….
There is (still) a general issue with Timeout’s in Ruby, relating to its use of Thread.kill etc, see this link for details.
Unfortunately its an issue in JRuby too. The solutions suggested in the link are quite low-level (and no other high option seems to be exposed).
I have some code that uses Net::HTTP that we need to set timeouts on (we need to know if the other end has a problem) which under load, hits the above issue – we start running out of resources (too many open files). Being a lazy/pragmatic programmer and having the advantage of working with JRuby – I decided to cheat and switch to a Java library that solves the timeout problem properly. Meet my friend, httpcomponents (formerly httpclient)
I tried to use it directly in Ruby, but that gets messy, so I wrapped it up a bit:
[code]
package com.x;
import org.apache.http.HttpEntity;
import org.apache.http.HttpHost;
import org.apache.http.HttpResponse;
import org.apache.http.auth.AuthScope;
import org.apache.http.auth.UsernamePasswordCredentials;
import org.apache.http.client.AuthCache;
import org.apache.http.client.methods.HttpPost;
import org.apache.http.client.protocol.ClientContext;
import org.apache.http.conn.params.ConnRoutePNames;
import org.apache.http.entity.StringEntity;
import org.apache.http.impl.auth.BasicScheme;
import org.apache.http.impl.client.BasicAuthCache;
import org.apache.http.impl.client.DefaultHttpClient;
import org.apache.http.params.CoreConnectionPNames;
import org.apache.http.params.HttpParams;
import org.apache.http.params.SyncBasicHttpParams;
import org.apache.http.protocol.BasicHttpContext;
import org.apache.http.util.EntityUtils;
import java.io.IOException;
import java.util.HashMap;
import java.util.Map;
/**
* User: kimptonc
* Date: 09/01/13
* Time: 12:05
* Thin wrapper around HttpClient lib – used by the ruby code
*/
public class HttpClient {
private String host;
private int port;
private String user;
private String pass;
private DefaultHttpClient httpclient;
private HttpHost targetHost;
private BasicAuthCache authCache;
private int conn_timeout;
private int so_timeout;
private static Object lock = new Object();
public HttpClient(String host, int port, String user, String pass, int conn_timeout, int so_timeout) {
this.host = host;
this.port = port;
this.user = user;
this.pass = pass;
this.conn_timeout = conn_timeout;
this.so_timeout = so_timeout;
while (httpclient == null)
{
synchronized (lock) {
connect();
}
}
}
public Map post(String url, String requestBody) throws Exception
{
try {
while (httpclient == null)
{
synchronized (lock) {
connect();
}
}
HttpPost post = new HttpPost(url);
StringEntity requestEntity = new StringEntity(requestBody);
requestEntity.setContentType(“application/xml”);
post.setEntity(requestEntity);
// Add AuthCache to the execution context
BasicHttpContext localcontext = new BasicHttpContext();
localcontext.setAttribute(ClientContext.AUTH_CACHE, authCache);
HttpResponse response = httpclient.execute(targetHost, post, localcontext);
HttpEntity entity = response.getEntity();
Map responseMap = new HashMap();
String responseBody = EntityUtils.toString(entity);
responseMap.put(“body”,responseBody);
responseMap.put(“status”,response.getStatusLine());
EntityUtils.consume(entity); // needed?
return responseMap;
} catch (Exception e) {
httpclient.getConnectionManager().shutdown();
httpclient = null;
throw e;
}
}
private void connect() {
if (httpclient != null)
return; // looks like we are connected, so ignore
targetHost = new HttpHost(host, port, “http”);
HttpParams params = new SyncBasicHttpParams();
params
.setIntParameter(CoreConnectionPNames.SO_TIMEOUT, so_timeout)
.setIntParameter(CoreConnectionPNames.CONNECTION_TIMEOUT, conn_timeout);
httpclient = new DefaultHttpClient(params);
httpclient.getCredentialsProvider().setCredentials(
new AuthScope(targetHost.getHostName(), targetHost.getPort()),
new UsernamePasswordCredentials(user, pass));
// Create AuthCache instance
authCache = new BasicAuthCache();
// Generate BASIC scheme object and add it to the local
// auth cache
BasicScheme basicAuth = new BasicScheme();
authCache.put(targetHost, basicAuth);
// for testing against a proxy in dev (eg Charles)
// HttpHost proxy = new HttpHost(“localhost”, 8888);
// httpclient.getParams().setParameter(ConnRoutePNames.DEFAULT_PROXY, proxy);
}
public static void main(String[] args) throws Exception {
HttpClient hc = new HttpClient(“localhost”, 8080,”one”, “two”,1,1);
Map resp = hc.post(“/v1/bond”,”<bond/>”);
System.out.println(“Resp:”+resp);
}
}
[/code]
Then to use it in the Ruby, code its just this:
[code]
@http_client = com.x.HttpClient.new(@svc_host, @svc_port.to_i,@svc_user,@svc_pass, open_timeout.to_i, read_timeout.to_i)
resp = @http_client.post @url_path, xml
[/code]
And it works – when we put this under load, we dont run out of resources :)
Scaling Rails…
For the last few weeks I have been trying to tune a rails app… getting to the point where we are wondering which way to scale – bigger box or more boxes…
I have tried to summarise the problem on Stackoverflow here.
If you have any ideas/suggestions, please add them there.
Thanks in advance :)
I created a (Ruby) DSL …
A DSL is a Domain Specific Language – that is, small language that should help frame solutions to particular problems.
In this case, we had some config for a report writer program that was parsed using custom code, things like so:
[code]
<report name>
{
FORMAT = CSV
RECORD = A.PATH.TO.RECORD
MODE = streaming
KEYFIELD = TradeId
FIELDS {
Id
TradeId
RecordType
Date
Time
}
WHERE Date = “20120924”
}
[/code]
The parser was getting hairier and we need to add more features – more complex WHERE clauses etc. So, it seemed an opportunity to throw in a DSL. This is the new DSL format for the above:
[code]
report “<report name>” do
format “CSV”
record “A.PATH.TO.RECORD”
mode “streaming”
keyfield “TradeId”
fields do
column “Id”
column “TradeId”
column “RecordType”
column “Date”
column “Time”
end
where do
Date == “20120924”
end
end
[/code]
When I first saw things like this, it seemed like magic – Ruby must be doing really complex stuff to handle it – but it isnt :)
Its all down to defining methods with the above names and handling the blocks passed to them.
For example there is a “report” method, like so:
[code]
def report(name, &block)
# save report name and &block of code for later use
end
[/code]
The “&block” bit is little funky – its a way of capturing the code passed between the “do … end” block above.
To handle the next level down – the code within the report block, I defined a class with those methods and the block is “call”‘d within the context of that block.
[code]
class WriterDefInRuby < Java::com.WriterDefBase # to make things fun, the class needs to implement a Java interface :)
def where(&block)
@where_clause = block
end
def setup(&block)
self.instance_eval &block
end
...
end
[/code]
So, the above class is used like this:
[code]
def report(name, &block)
writer_def = WriterDefInRuby.new
writer_def.name = name
writer_def.setup(&block)
end
[/code]
And thats it. We had a third level down for the fields stuff - but its just more of the same, another class, eval the block in the context of that class.
PS Thanks to several sites explaining DSL’s in much better detail and several example gems, like Rose and Docile