This is a very simple search engine web application which is somehow "specialized" in searching for financial news on the web. It's still very rudimental and far from retrieving good results, because I didn't have much time to develop and therefore to improve it. But my goal is to continue the development (especially of the missing but important features that I will mention next) of this search engine as soon as I've a little bit of time.
So far it only tries to find "quite blindly" (i.e. without a certain level of "intelligence") matches (with respect to the user's query) in the content of the webpages stored in the database (Hbase) without actually looking at the title or the URL or other characteristics of the webdocuments, such as anchor texts, language or format of the web document, etc. This is because I didn't have much time to learn how to use the API (i.e., Spring Data Solr) that allows the app to connect to the searching platform (i.e., Solr) and therefore how to perform more complex queries for more "expected" results.
It also doesn't allow you to search for exact matches or expressions, like
what's the meaning of "easy easy lemon squeezy?"
On ther other hand, you can try to write
easy easy lemon squeezy
and if there's a document in the database whose contents contain that sentence, it should be able to retrieve it, although it would not be able to distinguish very well which document is the best for the user's needs, which can vary a lot, of course. In general, this would be not easy to do even if I had a lot more time.
More specifically, the actual algorithm right now being used for searching is very simple and dumb. Given a set of words or terms ("eco"
was a term "eco"
, "economic"
or "economy"
would also be retrieved), ignoring any case sensitivity, i.e., "economic"
would be the same as "ECOnoMIC"
.
These documents were previously downloaded (and parsed) by a so-called crawler (in this case it was Nutch) and indexed (i.e., put simply, organized in a way for easy retrieval and ranking using specialized data structures such as inverted indices) by Solr, which under the hood uses another program called Lucene for indexing and searching.
I've not customized the indexing process of Solr (because of lack of time, of course). Moreover I didn't go much into the details of how actually Solr works, which exact approaches or data structures it uses, etc. I'm only aware of the fact that Lucene uses a vector space model and a boolean model to determine how relevant a given document is to a user's query, as it's explained here.
So, regarding the searching process there's much to improve in this application! Check the TODOs
section below for more details of features I would like to add to this SE.
Regarding the presentation of the results, a nice feature that's already implemented is the highlighting of the matches in the contents of the web documents in the result snippets (not in the title or in the URL). Other useful features that could improve the user experience are query suggestion and query correction, which I also want to implement. Right now the UI is very simple, but this was also my goal, even though, for example, I could have added other optional features which are very related to financial news, like displaying stock markets, etc.
I've decided to use Nutch version 2 because I was interested in having some flexibility in the choice of the data storage and to have some abstraction with respect to the crawling phase. Thus I also had to make a choice of the database: I chose Hbase mostly for it's strong connection with Nutch 2. These choice of course brough some tedious consequences, because I had to build Nutch from the source code.
Instead of using Lucene directly and having full freedom to design my app I decided to opt for the more "concrete" search platform Solr, which is built on top of Lucene and can run on the most used and famous Java servlet containers, such as Jetty or Tomcat. More specifically, since I decided to use SpringBoot as the web framework to develop the backend of my search web app (on top of Tomcat), I used the Spring Data Solr, which is a tool/library that provides already a few nice features to query Solr and thus provides another level of abstraction with respect to Solr, even though it's of course quite connected to it. These are one of the most interesting SpringBoot's features which ease a lot the query phase to a database, so why shouldn't I use it?
Regarding the configuration of Nutch and Solr, in particular the configuration of the schema, e.g. the configuration of the fields for each crawled and indexed document, I decided not mess much with it and to basically copy the apache-nutch-2.3.1/conf/schema.xml
file to the only Solr's core or collection (since they must be the same or at least compatible, lets say) that I'm using, that is solr-4.10.3/example/solr/collection1/conf/
.
I used Angular JS and Bootstrap to develop the simple front-end interface, because I wanted to get more into Angular JS and because I had already some knowledge of Bootstrap.
All of the following commands and explanations assume that you're on a unix-like system and that you have Java
and maven
installed. Currently I'm using Java 1.8.0_40
and Apache Maven 3.3.9
. If you're on Windows and you're stuck, feel free to contact me.
Software you will absolutely need to download is:
-
apache-nutch-2.3.1
(source distribution, since at the time of this writing there's still no binary one)
Other software you may eventually need, if you use the script ./build.sh
:
Notice that you may have already most of these programs installed on your system.
If you're on a Mac OS X, you can install most of these programs using for example the package manager Homebrew or port. On Linux the package manager may be different but the process should be the same.
I've created a script build.sh
which basically automates the tasks described in the sections configuration
and commands
. This script is still not robust, but in theory could avoid you to read the sections I've just mentioned and simply acess the web app at http://127.0.0.1:3000.
Before that though you may need to add the following lines to your /etc/hosts
file (as explained in the first tutorial mentioned below):
##
# Host Database
#
# localhost is used to configure the loopback interface
# when the system is booting. Do not change this entry.
##
127.0.0.1 localhost.localdomain localhost LMC-032857
::1 ip6-localhost ip6-loopback
fe80::1%lo0 ip6-localhost ip6-loopback
And you should also export the home folder of your JavaVM. Check the mentioned tutorial for more info.
After that, you can run the script as follows:
source ./build.sh
start
You can't simply do ./build.sh
because there's no function that's called directly, instead you have the freedom and flexibility to choose the one that you need. For example, you may just want to rebuild Nutch and pass the crawled data to Solr again. In that case, simply call:
nutch
provided that you have already executed source ./build.sh
. Check the source code of ./build.sh
to know more about the specific functions you can call.
Note:after the script finishes, you may want to crawl web documents. In that case, simply call the function nutch_update
by passing as first parameter the value for -topN
and as second parameter true
or false
depending if you want correspondingly to inject the seeds or not again.
These are tutorials that I encourage you to follow before proceeding with the next section, if you are not familiar with Nutch, HBase and Solr.
-
https://wiki.apache.org/nutch/NutchTutorial (for Nutch 1.X)
-
https://wiki.apache.org/nutch/Nutch2Tutorial (for Nutch 2.X)
Before running the web app, you first need to download apache-nutch-2.3.1
, hbase-0.98.8-hadoop2
and solr-4.10.3
(I encourage you to download all of the three programs to the folder where this README resides, i.e., at the same level as the folder app/
). It's possible that other versions of the above mentioned software also work together without problems of compatibility, but these versions are the suggested ones to work together by the Nutch's creators. I decided to pick the lastest version of Nutch (2.X), which is still distributed only as source code which can be compiled, because I was interested in having some flexibility in choosing a separate database to store the crawled data, which Nutch 2.X provides quite nicely.
Once you have those 3 programs, you should replace the files that you find in configuration/
for each of the programs above with their corresponding original version. If you don't do this, "you will probably not be able to run anything"... Note that I've left the structure of the folders the same as the original, but I kept only the required folders and files.
Please, do the following, first of all:
Add the to your /etc/hosts
file the following lines:
##
# Host Database
#
# localhost is used to configure the loopback interface
# when the system is booting. Do not change this entry.
##
127.0.0.1 localhost.localdomain localhost LMC-032857
::1 ip6-localhost ip6-loopback
fe80::1%lo0 ip6-localhost ip6-loopback
and export the home folder of your JavaVM. Check the first tutorial above, if you have any problems in doing one of these.
If you have already replaced the files I've mentioned in the section configuration
, then, in theory, you're able to build nutch from the source code. Before that, make sure you have ant
installed on your operating system. If you're on a Mac OS X and you have Homebrew
installed, you can simply do brew install ant
.
Once you have ant
installed, you can build nutch with the following commands:
ant clean
ant runtime
This should take less than 1 minute. Once that's finished, you need to start hbase
. Enter inside the folder hbase-0.98.8-hadoop2
and type:
bin/start-hbase.sh
Once that's done, you should also start the search platform solr
. To do that, go inside the folder you downloaded and unzipped solr-4.10.3
and type the following:
bin/solr start -c -dir example
You should see a message similar to:
Waiting to see Solr listening on port 8983 [/]
Started Solr server on port 8983 (pid=38649). Happy searching!
You can keep track of this pid=38649
and eventually kill the solr process when you're done with the searching with the following command:
kill 38649
After that you can finally crawl data (starting from the seeds/URLs that you should have copied to the nutch folder) and inject it into the database. To do it, run the following commands (in order):
-
Inject the seeds (i.e., initial URLs) into the database
runtime/local/bin/nutch inject seeds
-
Selects the top 1000 URLs
runtime/local/bin/nutch generate -topN 1000
-
Downloads the documents corresponding to the selected URLs
runtime/local/bin/nutch fetch -all
-
Parses the downloaded documents
runtime/local/bin/nutch parse -all
-
Updates the database with the new parsed documents
runtime/local/bin/nutch updatedb -all
Once that's finished, you can pass the crawled data to Solr to be indexed with the following command:
runtime/local/bin/nutch solrindex http://localhost:8983/solr -all
After all these steps, if you haven't encountered any problems, you can go to the next section and run the web app!
If you want to run this app from Intellij IDEA, you could have problems because of the resources managed by wro4j are not downloaded. I've not tried to run this app from Eclipse
, but apparently there's a plug-in for Eclipse for wro4j, so I guess you can safely run the app from Eclipse. Anyway, I advise you to use the following commands from the terminal to run the app, which must be run from inside the folder app/
.
mvn clean install
mvn spring-boot:run
After the execution of the command mvn spring-boot:run
, go to your favourite browser and type the following URL:
localhost:3000
This section describes some errors and exceptions that I've faced while setting up Nutch, HBase, Solr, etc. In general, I found the documentation lacking of many details and updates. I encountered many problems especially when trying to combine the tools because explanations are outdated or misleading.
indexer.IndexingJob - SolrIndexerJob: java.lang.RuntimeException: job failed: name=apache-nutch-2.3.1.jar, jobid=job_local441025530_0001
This java.lang.RuntimeException
exception can be raised for various reasons, so the only way to really understand the problem is to go into the compiled version of Nutch (specifically in apache-nutch-2.3.1/runtime/local/logs/
, open the current log file and check for the corresponding error.
java.lang.Exception: org.apache.solr.client.solrj.impl.HttpSolrServer$RemoteSolrException: ERROR: [doc=com.barchart.www:https/options] multiple values encountered for non multiValued field meta_keywords: [equity option quotes,options quotes,option prices,options chain,option chains,options trading, , options trading strategy,options trading strategies,calls,puts,call and put options,options screener,equity options,option bid ask,open interest,implied volatility,option greeks,option expiration date,stock options quotes,stock options, stock options trading,call option quotes,put option quotes,Stock Options news, Stock Options articles] at org.apache.hadoop.mapred.LocalJobRunner$Job.runTasks(LocalJobRunner.java:462) at org.apache.hadoop.mapred.LocalJobRunner$Job.run(LocalJobRunner.java:522) Caused by: org.apache.solr.client.solrj.impl.HttpSolrServer$RemoteSolrException: ERROR: [doc=com.barchart.www:https/options] multiple values encountered for non multiValued field meta_keywords: [equity option quotes, options quotes, option prices, options chain, option chains, options trading, options trading strategy, options trading strategies, calls, puts, call and put options, options screener, equity options, option bid ask, open interest, implied volatility, option greeks, option expiration date, stock options quotes, stock options, stock options trading, call option quotes, put option quotes, Stock Options news, Stock Options articles] at org.apache.solr.client.solrj.impl.HttpSolrServer.request(HttpSolrServer.java:491) at org.apache.solr.client.solrj.impl.HttpSolrServer.request(HttpSolrServer.java:197) at org.apache.solr.client.solrj.request.AbstractUpdateRequest.process(AbstractUpdateRequest.java:117) at org.apache.solr.client.solrj.SolrServer.add(SolrServer.java:68) at org.apache.solr.client.solrj.SolrServer.add(SolrServer.java:54) at org.apache.nutch.indexwriter.solr.SolrIndexWriter.write(SolrIndexWriter.java:84) at org.apache.nutch.indexer.IndexWriters.write(IndexWriters.java:84) at org.apache.nutch.indexer.IndexerOutputFormat$1.write(IndexerOutputFormat.java:48) at org.apache.nutch.indexer.IndexerOutputFormat$1.write(IndexerOutputFormat.java:43) at org.apache.hadoop.mapred.MapTask$NewDirectOutputCollector.write(MapTask.java:635) at org.apache.hadoop.mapreduce.task.TaskInputOutputContextImpl.write(TaskInputOutputContextImpl.java:89) at org.apache.hadoop.mapreduce.lib.map.WrappedMapper$Context.write(WrappedMapper.java:112) at org.apache.nutch.indexer.IndexingJob$IndexerMapper.map(IndexingJob.java:120) at org.apache.nutch.indexer.IndexingJob$IndexerMapper.map(IndexingJob.java:69) at org.apache.hadoop.mapreduce.Mapper.run(Mapper.java:145) at org.apache.hadoop.mapred.MapTask.runNewMapper(MapTask.java:764) at org.apache.hadoop.mapred.MapTask.run(MapTask.java:340) at org.apache.hadoop.mapred.LocalJobRunner$Job$MapTaskRunnable.run(LocalJobRunner.java:243) at java.util.concurrent.Executors$RunnableAdapter.call(Executors.java:511) at java.util.concurrent.FutureTask.run(FutureTask.java:266) at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1142) at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:617) at java.lang.Thread.run(Thread.java:745) 2016-11-29 13:34:02,501 ERROR indexer.IndexingJob - SolrIndexerJob: java.lang.RuntimeException: job failed: name=apache-nutch-2.3.1.jar, jobid=job_local441025530_0001 at org.apache.nutch.util.NutchJob.waitForCompletion(NutchJob.java:120) at org.apache.nutch.indexer.IndexingJob.run(IndexingJob.java:154) at org.apache.nutch.indexer.IndexingJob.index(IndexingJob.java:176) at org.apache.nutch.indexer.IndexingJob.run(IndexingJob.java:202) at org.apache.hadoop.util.ToolRunner.run(ToolRunner.java:70) at org.apache.nutch.indexer.IndexingJob.main(IndexingJob.java:211)
This problem is probably due to a connection problem to the database hbase or because the database was not started correctly... My advice is to restart whatever you were doing (i.e., call again start
). The process should delete the database and create another one...
-
Query suggestions
-
Query correction
-
Possibility to query for exact expression
-
Customize the searching process to be more "financial news-oriented".
-
Possibility to sort documents by title or url (for example)
-
...
© Nelson Brochado, 2016.