{-# LANGUAGE CPP #-} {-# LANGUAGE FlexibleContexts #-} {-# LANGUAGE ViewPatterns #-} {-# LANGUAGE PatternSynonyms #-} ----------------------------------------------------------------------------- -- | -- Module : Data.ByteString.Strict.Lens -- Copyright : (C) 2012-2016 Edward Kmett -- License : BSD-style (see the file LICENSE) -- Maintainer : Edward Kmett <ekmett@gmail.com> -- Stability : experimental -- Portability : non-portable -- ---------------------------------------------------------------------------- module Data.ByteString.Strict.Lens ( packedBytes, unpackedBytes, bytes , packedChars, unpackedChars, chars , pattern Bytes , pattern Chars ) where import Control.Lens import Control.Lens.Internal.ByteString import Data.ByteString (ByteString) import qualified Data.ByteString as Words import qualified Data.ByteString.Char8 as Char8 import Data.Word -- $setup -- >>> :set -XOverloadedStrings -- >>> import Control.Lens -- >>> import Numeric.Lens -- >>> import qualified Data.ByteString.Char8 as Char8 -- | 'Data.ByteString.pack' (or 'Data.ByteString.unpack') a list of bytes into a 'ByteString' -- -- @ -- 'packedBytes' ≡ 'from' 'unpackedBytes' -- 'Data.ByteString.pack' x ≡ x '^.' 'packedBytes' -- 'Data.ByteString.unpack' x ≡ x '^.' 'from' 'packedBytes' -- @ -- -- >>> [104,101,108,108,111]^.packedBytes -- "hello" packedBytes :: Iso' [Word8] ByteString packedBytes :: Iso' [Word8] ByteString packedBytes = ([Word8] -> ByteString) -> (ByteString -> [Word8]) -> Iso' [Word8] ByteString forall s a b t. (s -> a) -> (b -> t) -> Iso s t a b iso [Word8] -> ByteString Words.pack ByteString -> [Word8] Words.unpack {-# INLINE packedBytes #-} -- | 'Data.ByteString.unpack' (or 'Data.ByteString.pack') a 'ByteString' into a list of bytes -- -- @ -- 'unpackedBytes' ≡ 'from' 'packedBytes' -- 'Data.ByteString.unpack' x ≡ x '^.' 'unpackedBytes' -- 'Data.ByteString.pack' x ≡ x '^.' 'from' 'unpackedBytes' -- @ -- -- >>> "hello"^.packedChars.unpackedBytes -- [104,101,108,108,111] unpackedBytes :: Iso' ByteString [Word8] unpackedBytes :: Iso' ByteString [Word8] unpackedBytes = AnIso [Word8] [Word8] ByteString ByteString -> Iso' ByteString [Word8] forall s t a b. AnIso s t a b -> Iso b a t s from AnIso [Word8] [Word8] ByteString ByteString Iso' [Word8] ByteString packedBytes {-# INLINE unpackedBytes #-} -- | Traverse each 'Word8' in a 'ByteString'. -- -- This t'Traversal' walks the 'ByteString' in a tree-like fashion -- enable zippers to seek to locations in logarithmic time and accelerating -- many monoidal queries, but up to associativity (and constant factors) -- it is equivalent to the much slower: -- -- @ -- 'bytes' ≡ 'unpackedBytes' '.' 'traversed' -- @ -- -- >>> anyOf bytes (== 0x80) (Char8.pack "hello") -- False -- -- Note that when just using this as a t'Setter', @'setting' 'Data.ByteString.map'@ -- can be more efficient. bytes :: IndexedTraversal' Int ByteString Word8 bytes :: IndexedTraversal' Int ByteString Word8 bytes = p Word8 (f Word8) -> ByteString -> f ByteString IndexedTraversal' Int ByteString Word8 traversedStrictTree {-# INLINE bytes #-} -- | 'Data.ByteString.Char8.pack' (or 'Data.ByteString.Char8.unpack') a list of characters into a 'ByteString' -- -- When writing back to the 'ByteString' it is assumed that every 'Char' -- lies between @'\x00'@ and @'\xff'@. -- -- @ -- 'packedChars' ≡ 'from' 'unpackedChars' -- 'Data.ByteString.Char8.pack' x ≡ x '^.' 'packedChars' -- 'Data.ByteString.Char8.unpack' x ≡ x '^.' 'from' 'packedChars' -- @ -- -- >>> "hello"^.packedChars.each.re (base 16 . enum).to (\x -> if Prelude.length x == 1 then '0':x else x) -- "68656c6c6f" packedChars :: Iso' String ByteString packedChars :: Iso' String ByteString packedChars = (String -> ByteString) -> (ByteString -> String) -> Iso' String ByteString forall s a b t. (s -> a) -> (b -> t) -> Iso s t a b iso String -> ByteString Char8.pack ByteString -> String Char8.unpack {-# INLINE packedChars #-} -- | 'Data.ByteString.Char8.unpack' (or 'Data.ByteString.Char8.pack') a list of characters into a 'ByteString' -- -- When writing back to the 'ByteString' it is assumed that every 'Char' -- lies between @'\x00'@ and @'\xff'@. -- -- @ -- 'unpackedChars' ≡ 'from' 'packedChars' -- 'Data.ByteString.Char8.unpack' x ≡ x '^.' 'unpackedChars' -- 'Data.ByteString.Char8.pack' x ≡ x '^.' 'from' 'unpackedChars' -- @ -- -- >>> [104,101,108,108,111]^.packedBytes.unpackedChars -- "hello" unpackedChars :: Iso' ByteString String unpackedChars :: Iso' ByteString String unpackedChars = AnIso String String ByteString ByteString -> Iso' ByteString String forall s t a b. AnIso s t a b -> Iso b a t s from AnIso String String ByteString ByteString Iso' String ByteString packedChars {-# INLINE unpackedChars #-} -- | Traverse the individual bytes in a 'ByteString' as characters. -- -- When writing back to the 'ByteString' it is assumed that every 'Char' -- lies between @'\x00'@ and @'\xff'@. -- -- This t'Traversal' walks the 'ByteString' in a tree-like fashion -- enable zippers to seek to locations in logarithmic time and accelerating -- many monoidal queries, but up to associativity (and constant factors) -- it is equivalent to the much slower: -- -- @ -- 'chars' = 'unpackedChars' '.' 'traverse' -- @ -- -- >>> anyOf chars (== 'h') "hello" -- True chars :: IndexedTraversal' Int ByteString Char chars :: IndexedTraversal' Int ByteString Char chars = p Char (f Char) -> ByteString -> f ByteString IndexedTraversal' Int ByteString Char traversedStrictTree8 {-# INLINE chars #-} pattern Bytes :: [Word8] -> ByteString pattern $mBytes :: forall {r}. ByteString -> ([Word8] -> r) -> ((# #) -> r) -> r $bBytes :: [Word8] -> ByteString Bytes b <- (view unpackedBytes -> b) where Bytes [Word8] b = AReview ByteString [Word8] -> [Word8] -> ByteString forall b (m :: * -> *) t. MonadReader b m => AReview t b -> m t review AReview ByteString [Word8] Iso' ByteString [Word8] unpackedBytes [Word8] b pattern Chars :: String -> ByteString pattern $mChars :: forall {r}. ByteString -> (String -> r) -> ((# #) -> r) -> r $bChars :: String -> ByteString Chars b <- (view unpackedChars -> b) where Chars String b = AReview ByteString String -> String -> ByteString forall b (m :: * -> *) t. MonadReader b m => AReview t b -> m t review AReview ByteString String Iso' ByteString String unpackedChars String b