From 753eed65b8211b349f34119be80b41002e0d709f Mon Sep 17 00:00:00 2001
From: Demian Katz <demian.katz@villanova.edu>
Date: Wed, 29 May 2013 11:52:02 -0400
Subject: [PATCH] Added Solr 4 error listener.

---
 .../VuFind/Search/Solr/V4/ErrorListener.php   | 101 ++++++++++++++++
 .../response/solr/solr4-alphabrowse-error     |   9 ++
 .../fixtures/response/solr/solr4-parse-error  |   9 ++
 .../response/solr/solr4-undefined-field-error |   9 ++
 .../src/Search/Solr/V4/ErrorListenerTest.php  | 110 ++++++++++++++++++
 5 files changed, 238 insertions(+)
 create mode 100644 module/VuFind/src/VuFind/Search/Solr/V4/ErrorListener.php
 create mode 100644 module/VuFind/tests/fixtures/response/solr/solr4-alphabrowse-error
 create mode 100644 module/VuFind/tests/fixtures/response/solr/solr4-parse-error
 create mode 100644 module/VuFind/tests/fixtures/response/solr/solr4-undefined-field-error
 create mode 100644 module/VuFind/tests/unit-tests/src/Search/Solr/V4/ErrorListenerTest.php

diff --git a/module/VuFind/src/VuFind/Search/Solr/V4/ErrorListener.php b/module/VuFind/src/VuFind/Search/Solr/V4/ErrorListener.php
new file mode 100644
index 00000000000..f41d4ab249b
--- /dev/null
+++ b/module/VuFind/src/VuFind/Search/Solr/V4/ErrorListener.php
@@ -0,0 +1,101 @@
+<?php
+
+/**
+ * SOLR 4.x error listener.
+ *
+ * PHP version 5
+ *
+ * Copyright (C) Villanova University 2013.
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License version 2,
+ * as published by the Free Software Foundation.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; if not, write to the Free Software
+ * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
+ *
+ * @category VuFind2
+ * @package  Search
+ * @author   David Maus <maus@hab.de>
+ * @license  http://opensource.org/licenses/gpl-2.0.php GNU General Public License
+ * @link     http://vufind.org   Main Site
+ */
+
+namespace VuFind\Search\Solr\V4;
+
+use VuFindSearch\Backend\Exception\HttpErrorException;
+
+use Zend\EventManager\EventInterface;
+
+/**
+ * SOLR 3.x error listener.
+ *
+ * @category VuFind2
+ * @package  Search
+ * @author   David Maus <maus@hab.de>
+ * @license  http://opensource.org/licenses/gpl-2.0.php GNU General Public License
+ * @link     http://vufind.org   Main Site
+ */
+class ErrorListener
+{
+
+    /**
+     * Backends ot listen on.
+     *
+     * @var array
+     */
+    protected $backends;
+
+    /**
+     * Constructor.
+     *
+     * @param array $backends Name of backends to listen on
+     *
+     * @return void
+     */
+    public function __construct(array $backends)
+    {
+        $this->backends = $backends;
+    }
+
+    /**
+     * VuFindSearch.error
+     *
+     * @param EventInterface $event Event
+     *
+     * @return EventInterface
+     */
+    public function onSearchError(EventInterface $event)
+    {
+        $backend = $event->getParam('backend');
+        if (in_array($backend, $this->backends)) {
+            $error  = $event->getTarget();
+            if ($error instanceOf HttpErrorException) {
+                $body = $error->getResponse()->getBody();
+                $type = $error->getResponse()->getHeaders()->get('content-type')
+                    ->toString();
+                if (stristr($type, 'json')) {
+                    $body = json_decode($body);
+                    $reason = isset($body->error->msg) ? $body->error->msg : '';
+                } else if (stristr($type, 'xml')) {
+                    // TODO -- parse XML response
+                    $reason = '';
+                } else {
+                    $reason = '';
+                }
+                if (stristr($reason, 'org.apache.solr.search.SyntaxError')
+                    || stristr($reason, 'undefined field')
+                ) {
+                    $error->addTag('VuFind\Search\ParserError');
+                }
+            }
+        }
+        return $event;
+    }
+}
\ No newline at end of file
diff --git a/module/VuFind/tests/fixtures/response/solr/solr4-alphabrowse-error b/module/VuFind/tests/fixtures/response/solr/solr4-alphabrowse-error
new file mode 100644
index 00000000000..ed794c06e16
--- /dev/null
+++ b/module/VuFind/tests/fixtures/response/solr/solr4-alphabrowse-error
@@ -0,0 +1,9 @@
+HTTP/1.1 500 Server Error
+Cache-Control: no-cache, no-store
+Pragma: no-cache
+Expires: Sat, 01 Jan 2000 01:00:00 GMT
+Last-Modified: Wed, 29 May 2013 15:35:01 GMT
+ETag: "13ef0ed92c7"
+Content-Type: application/json; charset=UTF-8
+
+{"responseHeader":{"status":500,"QTime":135},"error":{"msg":"I couldn't find a browse index at: /home/dkatz/vufind3/solr/alphabetical_browse/author_browse.db.\nMaybe you need to create your browse indexes?","trace":"java.lang.Exception: I couldn't find a browse index at: /home/dkatz/vufind3/solr/alphabetical_browse/author_browse.db.\nMaybe you need to create your browse indexes?\n\tat au.gov.nla.solr.handler.HeadingsDB.openDB(BrowseRequestHandler.java:72)\n\tat au.gov.nla.solr.handler.HeadingsDB.reopenIfUpdated(BrowseRequestHandler.java:127)\n\tat au.gov.nla.solr.handler.Browse.reopenDatabasesIfUpdated(BrowseRequestHandler.java:553)\n\tat au.gov.nla.solr.handler.BrowseRequestHandler.handleRequestBody(BrowseRequestHandler.java:752)\n\tat org.apache.solr.handler.RequestHandlerBase.handleRequest(RequestHandlerBase.java:135)\n\tat org.apache.solr.core.SolrCore.execute(SolrCore.java:1817)\n\tat org.apache.solr.servlet.SolrDispatchFilter.execute(SolrDispatchFilter.java:639)\n\tat org.apache.solr.servlet.SolrDispatchFilter.doFilter(SolrDispatchFilter.java:345)\n\tat org.apache.solr.servlet.SolrDispatchFilter.doFilter(SolrDispatchFilter.java:141)\n\tat org.eclipse.jetty.servlet.ServletHandler$CachedChain.doFilter(ServletHandler.java:1307)\n\tat org.eclipse.jetty.servlet.ServletHandler.doHandle(ServletHandler.java:453)\n\tat org.eclipse.jetty.server.handler.ScopedHandler.handle(ScopedHandler.java:137)\n\tat org.eclipse.jetty.security.SecurityHandler.handle(SecurityHandler.java:560)\n\tat org.eclipse.jetty.server.session.SessionHandler.doHandle(SessionHandler.java:231)\n\tat org.eclipse.jetty.server.handler.ContextHandler.doHandle(ContextHandler.java:1072)\n\tat org.eclipse.jetty.servlet.ServletHandler.doScope(ServletHandler.java:382)\n\tat org.eclipse.jetty.server.session.SessionHandler.doScope(SessionHandler.java:193)\n\tat org.eclipse.jetty.server.handler.ContextHandler.doScope(ContextHandler.java:1006)\n\tat org.eclipse.jetty.server.handler.ScopedHandler.handle(ScopedHandler.java:135)\n\tat org.eclipse.jetty.server.handler.ContextHandlerCollection.handle(ContextHandlerCollection.java:255)\n\tat org.eclipse.jetty.server.handler.HandlerCollection.handle(HandlerCollection.java:154)\n\tat org.eclipse.jetty.server.handler.HandlerWrapper.handle(HandlerWrapper.java:116)\n\tat org.eclipse.jetty.server.Server.handle(Server.java:365)\n\tat org.eclipse.jetty.server.AbstractHttpConnection.handleRequest(AbstractHttpConnection.java:485)\n\tat org.eclipse.jetty.server.BlockingHttpConnection.handleRequest(BlockingHttpConnection.java:53)\n\tat org.eclipse.jetty.server.AbstractHttpConnection.headerComplete(AbstractHttpConnection.java:926)\n\tat org.eclipse.jetty.server.AbstractHttpConnection$RequestHandler.headerComplete(AbstractHttpConnection.java:988)\n\tat org.eclipse.jetty.http.HttpParser.parseNext(HttpParser.java:635)\n\tat org.eclipse.jetty.http.HttpParser.parseAvailable(HttpParser.java:235)\n\tat org.eclipse.jetty.server.BlockingHttpConnection.handle(BlockingHttpConnection.java:72)\n\tat org.eclipse.jetty.server.bio.SocketConnector$ConnectorEndPoint.run(SocketConnector.java:264)\n\tat org.eclipse.jetty.util.thread.QueuedThreadPool.runJob(QueuedThreadPool.java:608)\n\tat org.eclipse.jetty.util.thread.QueuedThreadPool$3.run(QueuedThreadPool.java:543)\n\tat java.lang.Thread.run(Thread.java:679)\n","code":500}}
diff --git a/module/VuFind/tests/fixtures/response/solr/solr4-parse-error b/module/VuFind/tests/fixtures/response/solr/solr4-parse-error
new file mode 100644
index 00000000000..712af02a5a8
--- /dev/null
+++ b/module/VuFind/tests/fixtures/response/solr/solr4-parse-error
@@ -0,0 +1,9 @@
+HTTP/1.1 400 Bad Request
+Cache-Control: no-cache, no-store
+Pragma: no-cache
+Expires: Sat, 01 Jan 2000 01:00:00 GMT
+Last-Modified: Wed, 29 May 2013 15:31:45 GMT
+ETag: "13ef0ea94ef"
+Content-Type: application/json; charset=UTF-8
+
+{"responseHeader":{"status":400,"QTime":2,"params":{"spellcheck":"true","facet":"true","sort":"score desc","facet.mincount":"1","spellcheck.q":"~~~","facet.limit":"30","hl.simple.pre":"{{{{START_HILITE}}}}","json.nl":"arrarr","hl.fl":"*","wt":"json","hl":"true","rows":"20","fl":"*,score","start":"0","facet.sort":"count","q":"((title_short:(\"~~~\")^750  OR  title_full_unstemmed:(\"~~~\")^600  OR  title_full_unstemmed:(~~~)^500  OR  title_full:(\"~~~\")^400  OR  title:(\"~~~\")^300  OR  title:(~~~)^250  OR  title_alt:(~~~)^200  OR  title_new:(~~~)^100)^50 OR series:(~~~)^50 OR series2:(~~~)^30 OR author:(\"~~~\")^300 OR author:(~~~)^250 OR author_fuller:(\"~~~\")^150 OR author_fuller:(~~~)^125 OR author2:(~~~)^50 OR author_additional:(~~~)^50 OR contents:(~~~)^10 OR topic_unstemmed:(\"~~~\")^550 OR topic_unstemmed:(~~~)^500 OR topic:(\"~~~\")^500 OR geographic:(\"~~~\")^300 OR genre:(\"~~~\")^300 OR allfields_unstemmed:(~~~)^10 OR fulltext_unstemmed:(~~~)^10 OR allfields:(~~~) OR fulltext:(~~~))","spellcheck.dictionary":"default","hl.simple.post":"{{{{END_HILITE}}}}","facet.field":["topic_facet","institution","building","format","callnumber-first","publishDate","authorStr","language","genre_facet","era_facet","geographic_facet"]}},"error":{"msg":"org.apache.solr.search.SyntaxError: Cannot parse '((title_short:(\"~~~\")^750  OR  title_full_unstemmed:(\"~~~\")^600  OR  title_full_unstemmed:(~~~)^500  OR  title_full:(\"~~~\")^400  OR  title:(\"~~~\")^300  OR  title:(~~~)^250  OR  title_alt:(~~~)^200  OR  title_new:(~~~)^100)^50 OR series:(~~~)^50 OR series2:(~~~)^30 OR author:(\"~~~\")^300 OR author:(~~~)^250 OR author_fuller:(\"~~~\")^150 OR author_fuller:(~~~)^125 OR author2:(~~~)^50 OR author_additional:(~~~)^50 OR contents:(~~~)^10 OR topic_unstemmed:(\"~~~\")^550 OR topic_unstemmed:(~~~)^500 OR topic:(\"~~~\")^500 OR geographic:(\"~~~\")^300 OR genre:(\"~~~\")^300 OR allfields_unstemmed:(~~~)^10 OR fulltext_unstemmed:(~~~)^10 OR allfields:(~~~) OR fulltext:(~~~))': Encountered \" <FUZZY_SLOP> \"~ \"\" at line 1, column 91.\nWas expecting one of:\n    <NOT> ...\n    \"+\" ...\n    \"-\" ...\n    <BAREOPER> ...\n    \"(\" ...\n    \"*\" ...\n    <QUOTED> ...\n    <TERM> ...\n    <PREFIXTERM> ...\n    <WILDTERM> ...\n    <REGEXPTERM> ...\n    \"[\" ...\n    \"{\" ...\n    <LPARAMS> ...\n    <NUMBER> ...\n    <TERM> ...\n    \"*\" ...\n    ","code":400}}
diff --git a/module/VuFind/tests/fixtures/response/solr/solr4-undefined-field-error b/module/VuFind/tests/fixtures/response/solr/solr4-undefined-field-error
new file mode 100644
index 00000000000..50eaf6d37a0
--- /dev/null
+++ b/module/VuFind/tests/fixtures/response/solr/solr4-undefined-field-error
@@ -0,0 +1,9 @@
+HTTP/1.1 400 Bad Request
+Cache-Control: no-cache, no-store
+Pragma: no-cache
+Expires: Sat, 01 Jan 2000 01:00:00 GMT
+Last-Modified: Wed, 29 May 2013 15:30:38 GMT
+ETag: "13ef0e9905e"
+Content-Type: application/json; charset=UTF-8
+
+{"responseHeader":{"status":400,"QTime":2,"params":{"spellcheck":"true","facet":"true","sort":"score desc","facet.mincount":"1","spellcheck.q":"test:test","facet.limit":"30","hl.simple.pre":"{{{{START_HILITE}}}}","json.nl":"arrarr","hl.fl":"*","wt":"json","hl":"true","rows":"20","fl":"*,score","start":"0","facet.sort":"count","q":"test:test","spellcheck.dictionary":"default","hl.simple.post":"{{{{END_HILITE}}}}","facet.field":["topic_facet","institution","building","format","callnumber-first","publishDate","authorStr","language","genre_facet","era_facet","geographic_facet"]}},"error":{"msg":"undefined field test","code":400}}
diff --git a/module/VuFind/tests/unit-tests/src/Search/Solr/V4/ErrorListenerTest.php b/module/VuFind/tests/unit-tests/src/Search/Solr/V4/ErrorListenerTest.php
new file mode 100644
index 00000000000..33dad92f826
--- /dev/null
+++ b/module/VuFind/tests/unit-tests/src/Search/Solr/V4/ErrorListenerTest.php
@@ -0,0 +1,110 @@
+<?php
+
+/**
+ * Unit tests for SOLR 3.x error listener.
+ *
+ * PHP version 5
+ *
+ * Copyright (C) Villanova University 2013.
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License version 2,
+ * as published by the Free Software Foundation.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; if not, write to the Free Software
+ * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
+ *
+ * @category VuFind2
+ * @package  Search
+ * @author   David Maus <maus@hab.de>
+ * @license  http://opensource.org/licenses/gpl-2.0.php GNU General Public License
+ * @link     http://vufind.org   Main Site
+ */
+
+namespace VuFindTest\Search\Solr\V4;
+
+use VuFind\Search\Solr\V4\ErrorListener;
+
+use VuFindSearch\Backend\Exception\HttpErrorException;
+
+use Zend\EventManager\Event;
+use Zend\Http\Response;
+
+use PHPUnit_Framework_TestCase as TestCase;
+
+use RuntimeException;
+
+/**
+ * Unit tests for SOLR 3.x error listener.
+ *
+ * @category VuFind2
+ * @package  Search
+ * @author   David Maus <maus@hab.de>
+ * @license  http://opensource.org/licenses/gpl-2.0.php GNU General Public License
+ * @link     http://vufind.org   Main Site
+ */
+class ErrorListenerTest extends TestCase
+{
+    /**
+     * Detect parser error response.
+     *
+     * @return void
+     */
+    public function testDetectParseError()
+    {
+        $response  = $this->createResponse('solr4-parse-error');
+
+        $exception = HttpErrorException::createFromResponse($response);
+        $params    = array('backend' => 'test');
+        $event     = new Event(null, $exception, $params);
+        $listener  = new ErrorListener(array('test'));
+        $listener->onSearchError($event);
+        $this->assertTrue($exception->hasTag('VuFind\Search\ParserError'));
+    }
+
+    /**
+     * Detect parser error response.
+     *
+     * @return void
+     */
+    public function testDetectUndefinedFieldError()
+    {
+        $response = $this->createResponse('solr4-undefined-field-error');
+
+        $exception = HttpErrorException::createFromResponse($response);
+        $params    = array('backend' => 'test');
+        $event     = new Event(null, $exception, $params);
+        $listener  = new ErrorListener(array('test'));
+        $listener->onSearchError($event);
+        $this->assertTrue($exception->hasTag('VuFind\Search\ParserError'));
+    }
+
+    /// Internal API
+
+    /**
+     * Return response fixture
+     *
+     * @param string $name Name of fixture
+     *
+     * @return Response Response
+     */
+    protected function createResponse($name)
+    {
+        $file = realpath(
+            \VUFIND_PHPUNIT_MODULE_PATH . '/fixtures/response/solr/' . $name
+        );
+        if (!$file) {
+            throw new RuntimeException(
+                sprintf('Unable to resolve fixture to fixture file: %s', $name)
+            );
+        }
+        $response = Response::fromString(file_get_contents($file));
+        return $response;
+    }
+}
\ No newline at end of file
-- 
GitLab