Skip to content

Commit

Permalink
Added code to resolve mkMessage should support context parsing #282 i…
Browse files Browse the repository at this point in the history
…ssue
  • Loading branch information
Oleksandr-Zhabenko committed Aug 22, 2024
1 parent 59ca619 commit 7201b20
Show file tree
Hide file tree
Showing 4 changed files with 91 additions and 41 deletions.
39 changes: 17 additions & 22 deletions .github/workflows/tests.yml
Original file line number Diff line number Diff line change
Expand Up @@ -14,34 +14,29 @@ jobs:
fail-fast: false
matrix:
os: [ubuntu-latest, macos-latest, windows-latest]
stack_args:
- --resolver=nightly
- --resolver=lts-21
- --resolver=lts-20
- --resolver=lts-19
- --resolver=lts-18
- --resolver=lts-16
- --resolver=lts-14
- --resolver=lts-12
- --stack-yaml=stack-ghc-9.2.yaml
exclude:
- os: windows-latest
stack_args: "--stack-yaml=stack-ghc-9.2.yaml"
args:
- "--resolver lts-22"
- "--resolver lts-21"
- "--resolver lts-20"

steps:
- name: Clone project
uses: actions/checkout@v2
uses: actions/checkout@v4

- name: Build and run tests
- name: Install stack if needed
shell: bash
run: |
set -ex
stack upgrade
stack --version
if [[ "${{ runner.os }}" = 'Windows' ]]
if [[ "${{ matrix.os }}" == "macos-latest" ]]
then
# Looks like a bug in Stack, this shouldn't break things
ls C:/ProgramData/Chocolatey/bin/
rm C:/ProgramData/Chocolatey/bin/ghc*
# macos-latest does not include Haskell tools as of 2024-05-06.
curl -sSL https://get.haskellstack.org/ | sh
fi
stack test --fast --no-terminal ${{ matrix.stack_args }}
- name: Build and run tests
shell: bash
run: |
set -ex
stack --version
stack test --fast --no-terminal ${{ matrix.args }}
76 changes: 61 additions & 15 deletions Text/Shakespeare/I18N.hs
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,7 @@
{-# LANGUAGE TypeFamilies #-}
{-# LANGUAGE MultiParamTypeClasses #-}
{-# LANGUAGE FlexibleInstances #-}
{-# LANGUAGE FlexibleContexts #-}
{-# LANGUAGE TypeSynonymInstances #-}
{-# LANGUAGE ExistentialQuantification #-}

Expand Down Expand Up @@ -62,22 +63,20 @@ module Text.Shakespeare.I18N
) where

import Language.Haskell.TH.Syntax hiding (makeRelativeToProject)
import Control.Applicative ((<$>))
import Control.Monad (filterM, forM)
import Data.Text (Text, pack, unpack)
import System.Directory
import Data.FileEmbed (makeRelativeToProject)
import Data.Maybe (catMaybes)
import Data.List (isSuffixOf, sortBy, foldl')
import Data.List (isSuffixOf, sortBy, foldl', intercalate)
import qualified Data.Map as Map
import qualified Data.ByteString as S
import Data.Text.Encoding (decodeUtf8)
import Data.Char (isSpace, toLower, toUpper)
import Data.Char (isSpace, toLower, toUpper, isLower)
import Data.Ord (comparing)
import Text.Shakespeare.Base (Deref (..), Ident (..), parseHash, derefToExp)
import Text.ParserCombinators.Parsec (parse, many, eof, many1, noneOf, (<|>))
import Text.ParserCombinators.Parsec (parse, many, eof, many1, noneOf, (<|>), string, spaces, char, option, alphaNum, sepBy1, try)
import Control.Arrow ((***))
import Data.Monoid (mempty, mappend)
import qualified Data.Text as T
import Data.String (IsString (fromString))

Expand Down Expand Up @@ -165,22 +164,69 @@ mkMessageCommon genType prefix postfix master dt rawFolder lang = do
Nothing -> error $ "Did not find main language file: " ++ unpack lang
Just def -> toSDefs def
mapM_ (checkDef sdef) $ map snd contents'
let mname = mkName $ dt ++ postfix
c1 <- fmap concat $ mapM (toClauses prefix dt) contents'
c2 <- mapM (sToClause prefix dt) sdef
let mname = mkName $ dt2 ++ postfix
c1 <- fmap concat $ mapM (toClauses prefix dt2 ) contents'
c2 <- mapM (sToClause prefix dt2) sdef
c3 <- defClause
return $
( if genType
then ((DataD [] mname [] Nothing (map (toCon dt) sdef) []) :)
then ((DataD [] mname [] Nothing (map (toCon dt2) sdef) []) :)
else id)
[ instanceD
[]
(ConT ''RenderMessage `AppT` (ConT $ mkName master) `AppT` ConT mname)
cxt -- Here the parsed context should be added, otherwise []
(ConT ''RenderMessage `AppT` (if ' ' `elem` master' then let (ts, us) = break (== ' ') . filter (\x -> x /= '(' && x /= ')') $ master' in let us1 = filter (/= ' ') us in ParensT (ConT (mkName ts) `AppT` VarT (mkName us1)) else ConT $ mkName master') `AppT` ConT mname)
[ FunD (mkName "renderMessage") $ c1 ++ c2 ++ [c3]
]
]

toClauses :: String -> String -> (Lang, [Def]) -> Q [Clause]
where (dt1, cxt0) = case (parse parseName "" dt) of
Left err -> error $ show err
Right x -> x
dt2 = concat . take 1 $ dt1
master' | cxt0 == [] = master
| otherwise = (\xss -> if length xss > 1 then '(':intercalate " " xss ++ ")" else concat . take 1 $ xss) . fst $
(case (parse parseName "" master) of
Left err -> error $ show err
Right x -> x)
cxt = fmap (\(c:rest) -> foldl' (\acc v -> acc `AppT` nameToType v) (ConT $ mkName c) rest) cxt0

nameToType :: String -> Type -- Is taken from the https://hackage.haskell.org/package/yesod-core-1.6.26.0/docs/src/Yesod.Routes.Parse.html#nameToType
nameToType t = if isTvar t
then VarT $ mkName t
else ConT $ mkName t

isTvar :: String -> Bool -- Is taken from the https://hackage.haskell.org/package/yesod-core-1.6.26.0/docs/src/Yesod.Routes.Parse.html#isTvar
isTvar (h:_) = isLower h
isTvar _ = False

parseName = do
cxt' <- option [] parseContext
args <- many parseWord
spaces
eof
return (args, cxt')

parseWord = do
spaces
many1 alphaNum

parseContext = try $ do
cxts <- parseParen parseContexts
spaces
_ <- string "=>"
return cxts

parseParen p = do
spaces
_ <- try ( char '(' )
r <- p
spaces
_ <- try ( char ')' )
return r

parseContexts =
sepBy1 (many1 parseWord) (spaces >> char ',' >> return ())

toClauses :: String -> String -> (Lang, [Def]) -> Q [Clause]
toClauses prefix dt (lang, defs) =
mapM go defs
where
Expand Down Expand Up @@ -255,13 +301,13 @@ conP name = ConP name []
conP = ConP
#endif

toCon :: String -> SDef -> Con
toCon :: String -> SDef -> Con
toCon dt (SDef c vs _) =
RecC (mkName $ "Msg" ++ c) $ map go vs
where
go (n, t) = (varName dt n, notStrict, ConT $ mkName t)

varName :: String -> String -> Name
varName :: String -> String -> Name
varName a y =
mkName $ concat [lower a, "Message", upper y]
where
Expand Down
3 changes: 1 addition & 2 deletions shakespeare.cabal
Original file line number Diff line number Diff line change
Expand Up @@ -146,8 +146,7 @@ test-suite test
, blaze-markup
, blaze-html
, exceptions



source-repository head
type: git
location: https://github.com/yesodweb/shakespeare.git
14 changes: 12 additions & 2 deletions test/Text/Shakespeare/I18NSpec.hs
Original file line number Diff line number Diff line change
@@ -1,16 +1,26 @@
{-# LANGUAGE MultiParamTypeClasses #-}
{-# LANGUAGE OverloadedStrings #-}
{-# LANGUAGE TemplateHaskell #-}
{-# LANGUAGE TypeFamilies #-}

module Text.Shakespeare.I18NSpec
( spec
) where

import Data.Text (Text)
import Text.Shakespeare.I18N
import Language.Haskell.TH.Syntax (Lift(..))

spec :: Monad m => m ()
spec = return ()

data Test = Test
class Lift master => YesodSubApp master where
data YesodSubAppData master :: *

newtype SubApp master = SubApp
{
getOrdering :: Ordering
}

mkMessage "(YesodSubApp master) => SubApp master" "test-messages" "en"

mkMessage "Test" "test-messages" "en"

0 comments on commit 7201b20

Please sign in to comment.