diff --git a/Makefile.am b/Makefile.am index d891a6d4b8d58568b45bf85bf3c75315f39ac993..d541fa25c7a13562ce7fce27b493ea4c4242e8ee 100644 --- a/Makefile.am +++ b/Makefile.am @@ -14,6 +14,9 @@ empty := space := $(empty) $(empty) comma := , +# Helper function to strip src/ and test/hs/ from a list +strip_hsroot = $(patsubst src/%,%,$(patsubst test/hs/%,%,$(1))) + # Use bash in order to be able to use pipefail SHELL=/bin/bash @@ -81,6 +84,9 @@ HS_DIRS = \ test/hs/Test/Ganeti/Hypervisor/Xen \ test/hs/Test/Ganeti/Query +# Haskell directories without the roots (src, test/hs) +HS_DIRS_NOROOT = $(filter-out src,$(filter-out test/hs,$(HS_DIRS))) + DIRS = \ $(HS_DIRS) \ autotools \ @@ -119,18 +125,7 @@ DIRS = \ ALL_APIDOC_HS_DIRS = \ $(APIDOC_HS_DIR) \ - $(APIDOC_HS_DIR)/Ganeti \ - $(APIDOC_HS_DIR)/Ganeti/Block \ - $(APIDOC_HS_DIR)/Ganeti/Block/Drbd \ - $(APIDOC_HS_DIR)/Ganeti/Confd \ - $(APIDOC_HS_DIR)/Ganeti/Curl \ - $(APIDOC_HS_DIR)/Ganeti/DataCollectors \ - $(APIDOC_HS_DIR)/Ganeti/HTools \ - $(APIDOC_HS_DIR)/Ganeti/HTools/Backend \ - $(APIDOC_HS_DIR)/Ganeti/HTools/Program \ - $(APIDOC_HS_DIR)/Ganeti/Hypervisor \ - $(APIDOC_HS_DIR)/Ganeti/Hypervisor/Xen \ - $(APIDOC_HS_DIR)/Ganeti/Query + $(patsubst %,$(APIDOC_HS_DIR)/%,$(call strip_hsroot,$(HS_DIRS_NOROOT))) BUILDTIME_DIR_AUTOCREATE = \ scripts \ @@ -611,6 +606,8 @@ HS_BUILT_SRCS_IN = \ $(patsubst %,%.in,$(filter-out src/Ganeti/Curl/Internal.hs,$(HS_BUILT_SRCS))) \ src/Ganeti/Curl/Internal.hsc +HS_LIBTESTBUILT_SRCS = $(HS_LIBTEST_SRCS) $(HS_BUILT_SRCS) + $(RUN_IN_TEMPDIR): | stamp-directories doc/html/index.html: ENABLE_MANPAGES = @@ -768,7 +765,7 @@ install-exec-hook: done endif -$(HS_ALL_PROGS): %: %.hs $(HS_LIBTEST_SRCS) $(HS_BUILT_SRCS) Makefile +$(HS_ALL_PROGS): %: %.hs $(HS_LIBTESTBUILT_SRCS) Makefile @if [ "$(notdir $@)" = "test" ] && [ "$(HS_NODEV)" ]; then \ echo "Error: cannot run unittests without the development" \ " libraries (see devnotes.rst)" 1>&2; \ @@ -1840,7 +1837,9 @@ py-apidoc: epydoc.conf $(RUN_IN_TEMPDIR) $(GENERATED_FILES) --output $(CURDIR)/$(APIDOC_PY_DIR) .PHONY: hs-apidoc -hs-apidoc: $(HS_BUILT_SRCS) +hs-apidoc: $(APIDOC_HS_DIR)/index.html + +$(APIDOC_HS_DIR)/index.html: $(HS_LIBTESTBUILT_SRCS) Makefile @test -n "$(HSCOLOUR)" || \ { echo 'HsColour' not found during configure; exit 1; } @test -n "$(HADDOCK)" || \ @@ -1853,25 +1852,26 @@ hs-apidoc: $(HS_BUILT_SRCS) $(LN_S) ../hscolour.css $(APIDOC_HS_DIR)/Ganeti/HTools/hscolour.css $(LN_S) ../hscolour.css $(APIDOC_HS_DIR)/Ganeti/Confd/hscolour.css set -e ; \ - cd src; \ - OPTGHC=""; \ + export LC_ALL=en_US.UTF-8; \ + OPTGHC="--optghc=-isrc --optghc=-itest/hs"; \ if [ "$(HS_PARALLEL3)" ]; \ then OPTGHC="$$OPTGHC --optghc=$(HS_PARALLEL3)"; \ fi; \ if [ "$(HS_REGEX_PCRE)" ]; \ then OPTGHC="$$OPTGHC --optghc=$(HS_REGEX_PCRE)"; \ fi; \ - RELSRCS="$(HS_LIB_SRCS:src/%=%) $(patsubst src/%,%,$(filter src/%,$(HS_BUILT_SRCS)))"; \ - for file in $$RELSRCS; do \ - hfile=`echo $$file|sed 's/\\.hs$$//'`.html; \ - $(HSCOLOUR) -css -anchor $$file > ../$(APIDOC_HS_DIR)/$$hfile ; \ + for file in $(HS_LIBTESTBUILT_SRCS); do \ + f_nosrc=$${file##src/}; \ + f_notst=$${f_nosrc##test/hs/}; \ + f_html=$${f_notst%%.hs}.html; \ + $(HSCOLOUR) -css -anchor $$file > $(APIDOC_HS_DIR)/$$f_html ; \ done ; \ - $(HADDOCK) --odir ../$(APIDOC_HS_DIR) --html --ignore-all-exports -w \ - -t ganeti -p haddock-prologue \ + $(HADDOCK) --odir $(APIDOC_HS_DIR) --html --ignore-all-exports -w \ + -t ganeti -p src/haddock-prologue \ --source-module="%{MODULE/.//}.html" \ --source-entity="%{MODULE/.//}.html#%{NAME}" \ $$OPTGHC \ - $(filter-out Ganeti/HTools/ExtLoader.hs,$(HS_LIB_SRCS:src/%=%)) + $(HS_LIBTESTBUILT_SRCS) .PHONY: TAGS TAGS: $(GENERATED_FILES) diff --git a/autotools/convert-constants b/autotools/convert-constants index e5296ea78ee80f6dc66213b7a5495640a906acfe..0cb07bbfc80bfab3acfddd29246b9bd21e5fa10e 100755 --- a/autotools/convert-constants +++ b/autotools/convert-constants @@ -182,7 +182,7 @@ def FormatDict(all_items, pfx_name, py_name, hs_name, mydict): # finally generate the output kv_pairs = ["(%s, %s)" % (k, v) for k, v in zip(key_v, val_v)] - return ["-- | Converted from Python dictionary %s" % py_name, + return ["-- | Converted from Python dictionary @%s@" % py_name, "%s :: [(%s, %s)]" % (hs_name, key_type, val_type), "%s = [%s]" % (hs_name, ", ".join(kv_pairs)), ] @@ -220,7 +220,7 @@ def ConvertVariable(prefix, name, value, all_items): elif hs_typeval is not None: # this is a simple value (hs_type, hs_val) = hs_typeval - lines.append("-- | Converted from Python constant %s" % fqn) + lines.append("-- | Converted from Python constant @%s@" % fqn) lines.append("%s :: %s" % (hs_name, hs_type)) lines.append("%s = %s" % (hs_name, hs_val)) elif isinstance(value, dict): @@ -245,7 +245,7 @@ def ConvertVariable(prefix, name, value, all_items): if compat.all(e is not None for e in tvs): ttypes = ", ".join(e[0] for e in tvs) tvals = FormatListElems(all_items, pfx_name, value, [e[1] for e in tvs]) - lines.append("-- | Converted from Python tuple %s" % fqn) + lines.append("-- | Converted from Python tuple @%s@" % fqn) lines.append("%s :: (%s)" % (hs_name, ttypes)) lines.append("%s = (%s)" % (hs_name, tvals)) else: @@ -264,7 +264,7 @@ def ConvertVariable(prefix, name, value, all_items): uniq_types = set(ttypes) if len(uniq_types) == 1: values = FormatListElems(all_items, pfx_name, value, tvals) - lines.append("-- | Converted from Python list or set %s" % fqn) + lines.append("-- | Converted from Python list or set @%s@" % fqn) lines.append("%s :: [%s]" % (hs_name, uniq_types.pop())) lines.append("%s = [%s]" % (hs_name, values)) else: @@ -274,7 +274,7 @@ def ConvertVariable(prefix, name, value, all_items): elif isinstance(value, RE_TYPE): tvs = HaskellTypeVal(value.pattern) assert tvs is not None - lines.append("-- | Converted from Python RE object %s" % fqn) + lines.append("-- | Converted from Python RE object @%s@" % fqn) lines.append("%s :: %s" % (hs_name, tvs[0])) lines.append("%s = %s" % (hs_name, tvs[1])) else: diff --git a/doc/devnotes.rst b/doc/devnotes.rst index 67fc06938514a34988062c9685ef432c2a42242f..941e54886e2f01f04d60f6b45ab5e6036b8dec46 100644 --- a/doc/devnotes.rst +++ b/doc/devnotes.rst @@ -29,14 +29,16 @@ Note that for pylint, at the current moment the following versions must be used:: $ pylint --version - pylint 0.21.1, - astng 0.20.1, common 0.50.3 + pylint 0.25.1, + astng 0.23.1, common 0.58.0 The same with pep8, other versions may give you errors:: $ pep8 --version 1.2 +Both these versions are the ones shipped with Debian Wheezy. + To generate unittest coverage reports (``make coverage``), `coverage <http://pypi.python.org/pypi/coverage>`_ needs to be installed. @@ -46,9 +48,9 @@ Installation of all dependencies listed here:: $ apt-get install pandoc python-epydoc graphviz $ cd / && sudo easy_install \ sphinx \ - logilab-astng==0.20.1 \ - logilab-common==0.50.3 \ - pylint==0.21.1 \ + logilab-astng==0.25.1 \ + logilab-common==0.58.0 \ + pylint==0.23.1 \ pep8==1.2 \ coverage diff --git a/src/Ganeti/Constants.hs.in b/src/Ganeti/Constants.hs.in index 44dd7448c6c4e1e36cf71ddcc1d5a39742de978a..7a52a9d30fcf7023fecc631294a26a71fc319673 100644 --- a/src/Ganeti/Constants.hs.in +++ b/src/Ganeti/Constants.hs.in @@ -1,14 +1,14 @@ {-| Ganeti constants. These are duplicated from the Python code. Note that this file is -autogenerated using autotools/convert_constants script with a header -from Constants.hs.in. +autogenerated using @autotools/convert_constants@ script with a header +from @Constants.hs.in@. -} {- -Copyright (C) 2009, 2010, 2011, 2012 Google Inc. +Copyright (C) 2009, 2010, 2011, 2012, 2013 Google Inc. This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by diff --git a/test/hs/Test/Ganeti/BasicTypes.hs b/test/hs/Test/Ganeti/BasicTypes.hs index a74bc299c1bf75a1fd10416156b161f0d120b72e..d365ab077a4a7795a94fba69c1640487741455be 100644 --- a/test/hs/Test/Ganeti/BasicTypes.hs +++ b/test/hs/Test/Ganeti/BasicTypes.hs @@ -7,7 +7,7 @@ {- -Copyright (C) 2009, 2010, 2011, 2012 Google Inc. +Copyright (C) 2009, 2010, 2011, 2012, 2013 Google Inc. This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by @@ -56,24 +56,31 @@ instance (Arbitrary a) => Arbitrary (Result a) where -- * Test cases --- | Tests the functor identity law (fmap id == id). +-- | Tests the functor identity law: +-- +-- > fmap id == id prop_functor_id :: Result Int -> Property prop_functor_id ri = fmap id ri ==? ri --- | Tests the functor composition law (fmap (f . g) == fmap f . fmap g). +-- | Tests the functor composition law: +-- +-- > fmap (f . g) == fmap f . fmap g prop_functor_composition :: Result Int -> Fun Int Int -> Fun Int Int -> Property prop_functor_composition ri (Fun _ f) (Fun _ g) = fmap (f . g) ri ==? (fmap f . fmap g) ri --- | Tests the applicative identity law (pure id <*> v = v). +-- | Tests the applicative identity law: +-- +-- > pure id <*> v = v prop_applicative_identity :: Result Int -> Property prop_applicative_identity v = pure id <*> v ==? v --- | Tests the applicative composition law (pure (.) <*> u <*> v <*> w --- = u <*> (v <*> w)). +-- | Tests the applicative composition law: +-- +-- > pure (.) <*> u <*> v <*> w = u <*> (v <*> w) prop_applicative_composition :: Result (Fun Int Int) -> Result (Fun Int Int) -> Result Int @@ -83,33 +90,47 @@ prop_applicative_composition u v w = v' = fmap apply v in pure (.) <*> u' <*> v' <*> w ==? u' <*> (v' <*> w) --- | Tests the applicative homomorphism law (pure f <*> pure x = pure (f x)). +-- | Tests the applicative homomorphism law: +-- +-- > pure f <*> pure x = pure (f x) prop_applicative_homomorphism :: Fun Int Int -> Int -> Property prop_applicative_homomorphism (Fun _ f) x = ((pure f <*> pure x)::Result Int) ==? pure (f x) --- | Tests the applicative interchange law (u <*> pure y = pure ($ y) <*> u). +-- | Tests the applicative interchange law: +-- +-- > u <*> pure y = pure ($ y) <*> u prop_applicative_interchange :: Result (Fun Int Int) -> Int -> Property prop_applicative_interchange f y = let u = fmap apply f -- need to extract the actual function from Fun in u <*> pure y ==? pure ($ y) <*> u --- | Tests the applicative\/functor correspondence (fmap f x = pure f <*> x). +-- | Tests the applicative\/functor correspondence: +-- +-- > fmap f x = pure f <*> x prop_applicative_functor :: Fun Int Int -> Result Int -> Property prop_applicative_functor (Fun _ f) x = fmap f x ==? pure f <*> x --- | Tests the applicative\/monad correspondence (pure = return and --- (<*>) = ap). +-- | Tests the applicative\/monad correspondence: +-- +-- > pure = return +-- +-- > (<*>) = ap prop_applicative_monad :: Int -> Result (Fun Int Int) -> Property prop_applicative_monad v f = let v' = pure v :: Result Int f' = fmap apply f -- need to extract the actual function from Fun in v' ==? return v .&&. (f' <*> v') ==? f' `ap` v' --- | Tests the monad laws (return a >>= k == k a, m >>= return == m, m --- >>= (\x -> k x >>= h) == (m >>= k) >>= h). +-- | Tests the monad laws: +-- +-- > return a >>= k == k a +-- +-- > m >>= return == m +-- +-- > m >>= (\x -> k x >>= h) == (m >>= k) >>= h prop_monad_laws :: Int -> Result Int -> Fun Int (Result Int) -> Fun Int (Result Int) @@ -122,7 +143,11 @@ prop_monad_laws a m (Fun _ k) (Fun _ h) = ((m >>= (\x -> k x >>= h)) ==? ((m >>= k) >>= h)) ] --- | Tests the monad plus laws ( mzero >>= f = mzero, v >> mzero = mzero). +-- | Tests the monad plus laws: +-- +-- > mzero >>= f = mzero +-- +-- > v >> mzero = mzero prop_monadplus_mzero :: Result Int -> Fun Int (Result Int) -> Property prop_monadplus_mzero v (Fun _ f) = printTestCase "mzero >>= f = mzero" ((mzero >>= f) ==? mzero) .&&.