Skip to content

Commit 0e6048f

Browse files
authored
Merge pull request #102 from matthewleon/nonemptyarray
conversions between NonEmptyArray/NonEmptyString
2 parents 62c9fdc + f2e40c0 commit 0e6048f

File tree

5 files changed

+85
-4
lines changed

5 files changed

+85
-4
lines changed

bench/Main.purs

+57
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,57 @@
1+
module Bench.Main where
2+
3+
import Prelude
4+
5+
import Control.Monad.Eff (Eff)
6+
import Control.Monad.Eff.Console (CONSOLE, log)
7+
import Data.Array.NonEmpty (fromArray)
8+
import Data.Maybe (fromJust)
9+
import Data.String (toCharArray)
10+
import Data.String.NonEmpty (fromFoldable1, fromNonEmptyCharArray)
11+
import Partial.Unsafe (unsafePartial)
12+
import Performance.Minibench (benchWith)
13+
14+
main :: Eff (console :: CONSOLE) Unit
15+
main = do
16+
log "NonEmpty conversions"
17+
log "======"
18+
log ""
19+
benchNonEmptyConversions
20+
21+
benchNonEmptyConversions :: Eff (console :: CONSOLE) Unit
22+
benchNonEmptyConversions = do
23+
log "fromNonEmptyCharArray: short"
24+
log "---"
25+
benchFromNonEmptyCharArray
26+
log ""
27+
28+
log "fromFoldable1"
29+
log "---"
30+
benchFromFoldable1
31+
log ""
32+
33+
where
34+
35+
benchFromNonEmptyCharArray = do
36+
log "short string"
37+
bench \_ -> fromNonEmptyCharArray shortStringArr
38+
39+
log "long string"
40+
bench \_ -> fromNonEmptyCharArray longStringArr
41+
42+
benchFromFoldable1 = do
43+
log "short string"
44+
bench \_ -> fromFoldable1 shortStringArr
45+
46+
log "long string"
47+
bench \_ -> fromFoldable1 longStringArr
48+
49+
shortStringArr = unsafePartial fromJust $ fromArray
50+
$ toCharArray "supercalifragilisticexpialidocious"
51+
longStringArr = unsafePartial fromJust $ fromArray
52+
$ toCharArray loremIpsum
53+
54+
bench = benchWith 100000
55+
56+
loremIpsum :: String
57+
loremIpsum = "Lorem ipsum dolor sit amet, consectetur adipiscing elit. Ut aliquet euismod ligula, vitae lacinia lorem imperdiet nec. Nulla volutpat ullamcorper mollis. Proin interdum quam a sem auctor, id tempus nisl pretium. Suspendisse potenti. Quisque ut libero consequat, suscipit sem a, malesuada nisi. Aliquam dictum odio mi, eu laoreet felis scelerisque non. Ut in odio vehicula, cursus augue sed, tincidunt lorem. Vestibulum consequat lectus eu commodo vulputate. Nam vitae faucibus ipsum. Curabitur sit amet neque sed est sagittis vehicula nec nec risus. Phasellus consectetur cursus malesuada. Vestibulum commodo lorem ut mauris mollis faucibus. Integer ut massa auctor, scelerisque nisi nec, rutrum nisl. Integer vel ex sem. Sed purus felis, molestie eget cursus vel, maximus ut augue. Curabitur nunc ligula, lobortis vitae vehicula a, volutpat nec sem. Phasellus non sapien ipsum. Mauris dolor justo, mollis at elit a, sollicitudin commodo quam. Curabitur posuere felis at nunc pharetra, eu convallis lectus dapibus. Aliquam ullamcorper porta fermentum. Donec at tellus metus. Donec pharetra tempor odio sit amet viverra. Nam vel metus libero. Vivamus maximus quis lacus id pharetra. Duis sed diam molestie, sodales leo id, pulvinar justo. In non augue tempor risus consectetur hendrerit. In libero nulla, elementum non ultrices eu, vehicula non ipsum. Maecenas in hendrerit tellus, sodales dignissim turpis. Ut odio diam, convallis in elit non, consequat gravida nisi. Cras egestas metus eleifend sapien efficitur, vel vulputate est porta. Aliquam posuere, magna nec bibendum luctus, quam risus efficitur sapien, id volutpat metus ex non lorem. Praesent velit eros, efficitur sed tortor quis, lobortis eleifend ligula. Sed tellus quam, aliquet vitae sagittis a, egestas eget massa. Etiam odio elit, hendrerit vel dui vel, fermentum pharetra neque. Curabitur quis mauris id lacus consectetur rhoncus non nec mauris. Mauris blandit tempor pretium. Donec non nisi finibus, lobortis dolor vitae, euismod arcu. Nullam scelerisque lacus in dolor volutpat mollis. Nunc vitae consectetur ligula, quis laoreet quam.Proin sit amet nisi eu orci hendrerit imperdiet vitae sit amet leo. Donec sodales id ante eget viverra. Nullam vitae elit in mauris accumsan feugiat id a velit. Nulla facilisi. Cras in turpis efficitur, consectetur justo quis, suscipit tortor. Sed tincidunt pellentesque sapien, in ultricies eros rhoncus sit amet. Integer blandit ornare lobortis. Duis dictum sit amet mauris sit amet cursus. Nullam nec nisl mauris. Praesent cursus imperdiet mi mattis luctus. Donec in tortor fermentum, efficitur turpis vel, facilisis augue. Integer egestas nisl et magna volutpat ornare. Donec pulvinar risus elit, eget viverra est feugiat in.Ut nec ante vestibulum neque pulvinar pretium sit amet eu nisi. Aliquam erat volutpat. Maecenas egestas nisi et mi congue, sed ultricies nibh posuere. Suspendisse potenti. Donec a nulla et velit elementum pretium. Pellentesque gravida imperdiet sem et varius. Praesent ac diam diam. Donec iaculis risus ex, ac eleifend sapien luctus ut. Fusce aliquet, lacus tincidunt porta malesuada, massa augue commodo nulla, ac malesuada tortor est sed eros. Praesent mattis, nisi eget ullamcorper vestibulum, lacus ante placerat metus, ac ullamcorper ante tellus vel nulla. Praesent vehicula in est sit amet varius. Sed facilisis felis sed sem porttitor rutrum. Etiam sollicitudin erat neque, id gravida metus scelerisque quis. Proin venenatis pharetra lectus ac auctor."

bower.json

+2-1
Original file line numberDiff line numberDiff line change
@@ -27,6 +27,7 @@
2727
},
2828
"devDependencies": {
2929
"purescript-assert": "#compiler/0.12",
30-
"purescript-console": "#compiler/0.12"
30+
"purescript-console": "#compiler/0.12",
31+
"purescript-minibench": "#compiler/0.12"
3132
}
3233
}

package.json

+4-1
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,10 @@
33
"scripts": {
44
"clean": "rimraf output && rimraf .pulp-cache",
55
"build": "eslint src && pulp build -- --censor-lib --strict",
6-
"test": "pulp test"
6+
"test": "pulp test",
7+
"bench:build": "purs compile 'bench/**/*.purs' 'src/**/*.purs' 'bower_components/*/src/**/*.purs'",
8+
"bench:run": "node --expose-gc -e 'require(\"./output/Bench.Main/index.js\").main()'",
9+
"bench": "npm run bench:build && npm run bench:run"
710
},
811
"devDependencies": {
912
"eslint": "^3.17.1",

src/Data/String/NonEmpty.purs

+12
Original file line numberDiff line numberDiff line change
@@ -10,12 +10,14 @@ module Data.String.NonEmpty
1010
, fromString
1111
, unsafeFromString
1212
, fromCharArray
13+
, fromNonEmptyCharArray
1314
, singleton
1415
, cons
1516
, snoc
1617
, fromFoldable1
1718
, toString
1819
, toCharArray
20+
, toNonEmptyCharArray
1921
, charAt
2022
, charCodeAt
2123
, toChar
@@ -52,6 +54,8 @@ module Data.String.NonEmpty
5254

5355
import Prelude
5456

57+
import Data.Array.NonEmpty (NonEmptyArray)
58+
import Data.Array.NonEmpty as NEA
5559
import Data.Foldable (class Foldable)
5660
import Data.Foldable as F
5761
import Data.Maybe (Maybe(..), fromJust)
@@ -60,6 +64,7 @@ import Data.Semigroup.Foldable as F1
6064
import Data.String (Pattern(..))
6165
import Data.String as String
6266
import Data.String.Unsafe as U
67+
import Partial.Unsafe (unsafePartial)
6368
import Unsafe.Coerce (unsafeCoerce)
6469

6570
-- | A string that is known not to be empty.
@@ -110,6 +115,9 @@ fromCharArray = case _ of
110115
[] -> Nothing
111116
cs -> Just (NonEmptyString (String.fromCharArray cs))
112117

118+
fromNonEmptyCharArray :: NonEmptyArray Char -> NonEmptyString
119+
fromNonEmptyCharArray = unsafePartial fromJust <<< fromCharArray <<< NEA.toArray
120+
113121
-- | Creates a `NonEmptyString` from a character.
114122
singleton :: Char -> NonEmptyString
115123
singleton = NonEmptyString <<< String.singleton
@@ -181,6 +189,10 @@ toChar (NonEmptyString s) = String.toChar s
181189
toCharArray :: NonEmptyString -> Array Char
182190
toCharArray (NonEmptyString s) = String.toCharArray s
183191

192+
-- | Converts the `NonEmptyString` into a non-empty array of characters.
193+
toNonEmptyCharArray :: NonEmptyString -> NonEmptyArray Char
194+
toNonEmptyCharArray = unsafePartial fromJust <<< NEA.fromArray <<< toCharArray
195+
184196
-- | Appends a string to this non-empty string. Since one of the strings is
185197
-- | non-empty we know the result will be too.
186198
-- |

test/Test/Data/String/NonEmpty.purs

+10-2
Original file line numberDiff line numberDiff line change
@@ -2,12 +2,13 @@ module Test.Data.String.NonEmpty (testNonEmptyString) where
22

33
import Data.String.NonEmpty
44

5-
import Effect (Effect)
6-
import Effect.Console (log)
5+
import Data.Array.NonEmpty as NEA
76
import Data.Array.Partial as AP
87
import Data.Foldable (class Foldable, foldl)
98
import Data.Maybe (Maybe(..), fromJust, isNothing, maybe)
109
import Data.Semigroup.Foldable (class Foldable1, foldMap1Default)
10+
import Effect (Effect)
11+
import Effect.Console (log)
1112
import Partial.Unsafe (unsafePartial)
1213
import Prelude (class Functor, Ordering(..), Unit, append, discard, negate, not, ($), (&&), (/=), (==))
1314
import Test.Assert (assert)
@@ -22,6 +23,9 @@ testNonEmptyString = do
2223
assert $ fromCharArray [] == Nothing
2324
assert $ fromCharArray ['a', 'b'] == Just (nes "ab")
2425

26+
log "fromNonEmptyCharArray"
27+
assert $ fromNonEmptyCharArray (NEA.singleton 'b') == singleton 'b'
28+
2529
log "singleton"
2630
assert $ singleton 'a' == nes "a"
2731

@@ -64,6 +68,10 @@ testNonEmptyString = do
6468
assert $ toCharArray (nes "ab") == ['a', 'b']
6569
assert $ toCharArray (nes "Hello☺\n") == ['H','e','l','l','o','☺','\n']
6670

71+
log "toNonEmptyCharArray"
72+
assert $ toNonEmptyCharArray (nes "ab")
73+
== unsafePartial fromJust (NEA.fromArray ['a', 'b'])
74+
6775
log "appendString"
6876
assert $ appendString (nes "Hello") " world" == nes "Hello world"
6977
assert $ appendString (nes "Hello") "" == nes "Hello"

0 commit comments

Comments
 (0)