brainsteam.co.uk/brainsteam/content/posts/legacy/2015-07-15-sssplit-improvem...

71 lines
7.8 KiB
Markdown
Raw Permalink Normal View History

2020-12-28 11:39:11 +00:00
---
author: James
2023-07-09 11:34:44 +01:00
date: 2015-07-15 19:33:29+00:00
post_meta:
- date
2020-12-28 11:39:11 +00:00
tags:
2023-07-09 11:34:44 +01:00
- phd
- demo
- partridge
- python
- sapienta
title: SSSplit Improvements
type: posts
url: /2015/07/15/sssplit-improvements/
2020-12-28 11:39:11 +00:00
---
2023-07-09 11:34:44 +01:00
2020-12-28 11:39:11 +00:00
## Introduction
As part of my continuing work on [Partridge][1], I’ve been working on improving the sentence splitting capability of SSSplit – the component used to split academic papers from PLosOne and PubMedCentral into separate sentences.
Papers arrive in our system as big blocks of text with the occasional diagram, formula or diagram and in order to apply CoreSC annotations to the sentences we need to know where each sentence starts and ends. Of course that means we also have to take into account the other &#8216;stuff&#8217; (listed above) floating around in the documents too. We can&#8217;t just ignore formulae and citations &#8211; they&#8217;re pretty important! That&#8217;s what SSSplit does. It carves up papers into sentence (_<s>_) elements whilst also leaving the XML structure of the rest of the document in tact.
The original SSSplit utility was written a number of years ago in Java and used Regular Expressions to parse XML (something that readers of [this StackOverflow article][2] will already know has a propensity to summon eldrich abominations from the otherworld). Due to the complex regular expressions, the old splitter was not particularly performant . Especially given the complex nature of some of the expressions (if you&#8217;re interested, check out one of the _simpler_ ones [here][3]).
Now, I can definitely see what the original authors were going for here. Regular expressions are very good for splitting sentences but not sentences inside complex XML documents. XML parsers are not particularly good for splitting sentences but are obviously good at parsing XML. I also understand that the original splitter was designed and then new bits glued on to make it suitable for new and different standards of XML leading to the gargantuan expressions like the one linked to above. I think they did a pretty good job given the information available to them at the time of writing.
I decided that the splitter needed a rewrite and went straight to my comfort zone to get it done: Python. I&#8217;m very familiar with the language &#8211; to the point now that I can write a fairly complicated program in it in a day if I&#8217;ve had enough coffee and sugar.
## Writing SSSplit 2.0
I decided that we needed to try and minimise excessive uses of regular expressions for both performance and maintainence/readability reasons.  I decided to try and do as much of the parsing of the document structure as possible using a traditional XML parser. I&#8217;d heard good things about [etree][4] which is part of the standard Python library and provides an informal dom-like interface. I used etree to inspect what I dubbed &#8216;P-level&#8217; xml elements first. These are elements that I consider to be at a &#8220;paragraph&#8221; level. Any sentences inside these elements are completely contained &#8211; they do not cross the boundaries into the next container (unless the author is a poet/fiction writer/doesn&#8217;t do English very well I think its a safe bet that they wouldn&#8217;t finish a paragraph mid-sentence). Within the p-level containers, I sweep for any sort of XML node &#8211; we&#8217;re interested in text nodes but also any sort of formatting like bold (<b>) elements.
When a text node is encountered, that&#8217;s when regular expressions start to kick in. We do a very simple match for punctuation just in front of a space and a capital letter and run it over the text node &#8211; these are &#8220;potential&#8221; splits. We also look for full stops at the very end of the text.
<pre lang="python">pattern = re.compile('(\.|\?|\!)(?=\s*[A-Z0-9$])|\.$')
m = pattern.search(txt)
</pre>
Of course this generates lots of false positives &#8211; what if we&#8217;ve found a decimal point inside a number? What if it&#8217;s an abbreviation like e.g. or i.e. or an initial like J. Ravenscroft? There is another regular expression check for decimal points and the string around the punctuation is checked against a list of common abbreviations. There&#8217;s also a list of authors both the writers of the paper in question and those who are cited in the paper too. The function checks that the full stop is not part of one of these authors&#8217; names.
There&#8217;s an important factor to remember: Text node does not imply finished sentence &#8211; they are interspersed with formulae and references as explained above. Therefore we can&#8217;t just finish the current sentence when we reach the end of a text node &#8211; only when we encounter a full stop (not part of an abbreviation or number), question mark or explanation mark. We also know that we can complete the current sentence at the end of a p-level container as I explained above.
Every time we start parsing a sentence, text nodes and other &#8216;stuff&#8217; deemed to be inside that sentence is accumulated into a list. Once we encounter the end of the sentence, the list is glued together and turned into an XML <s> element.
The next step was to see how effective the new splitter was against the old splitter and also manual annotation by professional scientific literature readers.
## Testing the splitter
To test the system I originally wrote a simple script that takes a set of manually annotated papers &#8211; strips them of their annotations so that the new splitter doesn&#8217;t get any clues &#8211; runs the new routine over them and then compares the output. This was very rudimentary as I was in a rush and didn&#8217;t tell me much about the success rate of my splitter. It did display the first and last words of each &#8220;detected&#8221; sentence for both manual and automatic annotation so I could at least see how well (if at all) the two lined up. I had to run the script on a paper-by-paper basis.
I managed to get the splitter working really well on a number of papers (we&#8217;re talking a 100% match) using this tool. However I realised that the majority of papers were still not being matched and it was becoming more and more of a chore to find which ones weren&#8217;t matching.
That&#8217;s why I decided to write a web-based visualisation tool for checking the splitter. The idea is that it runs on all papers giving an overall percentage of how well the automated splitter is working vs the manual splitter but also gives a per-paper figure. If you want to see which papers the system is really struggling with you can inspect them by clicking on them. This brings up a list of all the sentences and whether or not they align.
The tool is pretty useful as it gives me a clue as to which papers I need to tune the splitter with next.
Here&#8217;s a quick demo video of me using the tool to find papers that don&#8217;t match very well.
<div class="jetpack-video-wrapper">
<span class="embed-youtube" style="text-align:center; display: block;"><iframe class='youtube-player' width='660' height='372' src='https://www.youtube.com/embed/o1EpJ_zJcno?version=3&#038;rel=1&#038;showsearch=0&#038;showinfo=1&#038;iv_load_policy=1&#038;fs=1&#038;hl=en-US&#038;autohide=2&#038;wmode=transparent' allowfullscreen='true' style='border:0;' sandbox='allow-scripts allow-same-origin allow-popups allow-presentation'></iframe></span>
</div>
## Next steps
A lot of tuning has been done on how this system works but there&#8217;s still a long way to go yet. I&#8217;ll probably post another article talking about what further changes had to be made to make the parser effective!
[1]: http://papro.org.uk
[2]: http://stackoverflow.com/questions/1732348/regex-match-open-tags-except-xhtml-self-contained-tags/1732454#1732454
[3]: https://www.debuggex.com/r/vEyxqRg6xgN9ui_P
[4]: https://docs.python.org/2/library/xml.etree.elementtree.html