diff options
Diffstat (limited to 'test')
-rw-r--r-- | test/Makefile | 173 | ||||
-rw-r--r-- | test/core_app.mk | 1398 | ||||
-rw-r--r-- | test/core_compat.mk | 221 | ||||
-rw-r--r-- | test/core_plugins.mk | 86 | ||||
-rw-r--r-- | test/core_upgrade.mk | 124 | ||||
-rw-r--r-- | test/plugin_bootstrap.mk | 155 |
6 files changed, 2115 insertions, 42 deletions
diff --git a/test/Makefile b/test/Makefile index 59063e6..ba612fa 100644 --- a/test/Makefile +++ b/test/Makefile @@ -1,72 +1,161 @@ +# Copyright (c) 2015, Loïc Hoguin <[email protected]> # Copyright (c) 2014, Viktor Söderqvist <[email protected]> # This file is part of erlang.mk and subject to the terms of the ISC License. -# Tests for erlang.mk targets. If any test fails or if you run a target other -# than 'all', you must probably do 'make clean' before you can test again. +# Temporary application name, taken from rule name. + +APP = $(subst -,_,$@) +APP_TO_CLEAN = $(subst -,_,$(patsubst clean-%,%,$@)) + +# Erlang, quickly! + +ERL = erl +A0 -noinput -boot start_clean + +# Platform, condensed version. + +ifeq ($(shell uname -o),Msys) + PLATFORM = msys2 +else + PLATFORM = unix +endif + +# OTP master, for downloading files for testing. + +OTP_MASTER = https://raw.githubusercontent.com/erlang/otp/master # Verbosity. +# +# V=0: Show info messages only. +# V=1: Show test commands. +# V=2: Also show normal Erlang.mk output. +# V=3: Also show verbose Erlang.mk output. V ?= 0 -# Temporary files directory. +# t: Verbosity control for tests. +# v: Verbosity control for erlang.mk. +# i: Command to display (or suppress) info messages. -ERLANG_MK_TMP=$(CURDIR)/tmp -export ERLANG_MK_TMP - -# t = Verbosity control for tests -# v = Verbosity control for erlang.mk -# i = Command to display (or suppress) info messages ifeq ($V,0) - # Show info messages only t = @ v = V=0 >/dev/null 2>&1 - i = @echo + i = @echo $@: else ifeq ($V,1) - # Show test commands t = v = V=0 >/dev/null 2>&1 - i = @echo == + i = @echo == $@: else ifeq ($V,2) - # Show briefly what erlang.mk is doing t = @echo " TEST " $@; v = V=0 - i = @echo == + i = @echo == $@: else - # Show all commands with maximum verbosity t = v = V=1 - i = @echo == + i = @echo == $@: endif -.PHONY: all clean app ct eunit tests-cover docs +# Main targets. -.NOTPARALLEL: +.PHONY: all clean build -all: clean app ct eunit tests-cover docs pkgs - $i '+---------------------+' - $i '| All tests passed. |' - $i '+---------------------+' +all:: core -clean: - $t rm -rf app1 pkgs.log $(ERLANG_MK_TMP) +clean:: clean-core + $t rm -f erl_crash.dump -app: app1 - $i "app: Testing the 'app' target." - $t $(MAKE) -C app1 app $v - $i "Checking the modules line in the generated .app file." - $t [ `grep -E "{modules, *\['m'\]}" app1/ebin/app1.app | wc -l` -eq 1 ] - $t [ -e app1/ebin/m.beam ] - $i "Checking that '$(MAKE) clean-app' deletes ebin." - $t $(MAKE) -C app1 clean-app $v - $t [ ! -e app1/ebin ] - $i "Checking that '$(MAKE) app' returns non-zero on compile errors." - $t printf "%s\n" \ - "-module(syntax_error)." \ - "foo lorem_ipsum dolor sit amet." \ - > app1/src/syntax_error.erl - $t if $(MAKE) -C app1 app $v ; then false ; fi - $t rm app1/src/syntax_error.erl - $i "Test 'app' passed." +build: + $i "Generate a bleeding edge Erlang.mk" + $t cd .. && $(MAKE) $v + +# Core. + +.PHONY: core clean-core + +define include_core +core:: core-$1 +clean-core:: clean-core-$1 + +include core_$1.mk + +endef + +$(eval $(foreach t,$(patsubst %.mk,%,$(patsubst core_%,%,$(wildcard core_*.mk))),$(call include_core,$t))) + +# Plugins. + +define include_plugin +all:: $1 +clean:: clean-$1 + +include plugin_$1.mk + +endef + +$(eval $(foreach t,$(patsubst %.mk,%,$(patsubst plugin_%,%,$(wildcard plugin_*.mk))),$(call include_plugin,$t))) + +# Tests that don't easily fit into other categories. + +core:: core-clean-crash-dump core-distclean-tmp core-help +clean-core:: clean-core-clean-crash-dump clean-core-distclean-tmp clean-core-help + +.PHONY: core-clean-crash-dump core-distclean-tmp core-help clean-core-clean-crash-dump clean-core-distclean-tmp clean-core-help + +clean-core-clean-crash-dump clean-core-distclean-tmp clean-core-help: + $t rm -rf $(APP_TO_CLEAN)/ + +core-clean-crash-dump: build clean-core-clean-crash-dump + + $i "Bootstrap a new OTP library named $(APP)" + $t mkdir $(APP)/ + $t cp ../erlang.mk $(APP)/ + $t $(MAKE) -C $(APP) -f erlang.mk bootstrap-lib $v + + $i "Create a fake erl_crash.dump file" + $t touch $(APP)/erl_crash.dump + + $i "Clean the application" + $t $(MAKE) -C $(APP) clean $v + + $i "Check that the crash dump is removed" + $t test ! -e $(APP)/erl_crash.dump + +core-distclean-tmp: build clean-core-distclean-tmp + + $i "Bootstrap a new OTP application named $(APP)" + $t mkdir $(APP)/ + $t cp ../erlang.mk $(APP)/ + $t $(MAKE) -C $(APP) -f erlang.mk bootstrap all $v + + $i "Check that a .erlang.mk directory exists" + $t test -d $(APP)/.erlang.mk + + $i "Distclean the application" + $t $(MAKE) -C $(APP) distclean $v + + $i "Check if .erlang.mk directory got removed" + $t test ! -e $(APP)/.erlang.mk + +core-help: build clean-core-help + + $i "Bootstrap a new OTP library named $(APP)" + $t mkdir $(APP)/ + $t cp ../erlang.mk $(APP)/ + $t $(MAKE) -C $(APP) -f erlang.mk bootstrap-lib $v + + $i "Run 'make help' and check that it prints help" + $t test -n "`$(MAKE) -C $(APP) help` | grep Usage" + +# Legacy tests. +# +# The following tests are slowly being converted. +# Do NOT use -j with legacy tests. + +.PHONY: legacy clean-legacy ct eunit tests-cover docs + +legacy: clean-legacy ct eunit tests-cover docs pkgs + +clean-legacy: + $t rm -rf app1 pkgs.log ct: app1 $i "ct: Testing ct and related targets." diff --git a/test/core_app.mk b/test/core_app.mk new file mode 100644 index 0000000..55bf7b2 --- /dev/null +++ b/test/core_app.mk @@ -0,0 +1,1398 @@ +# Core: Building applications. + +CORE_APP_CASES = asn1 auto-git-id erlc-exclude erlc-opts erlc-opts-filter error generate-erl generate-erl-include generate-erl-prepend hrl hrl-recursive mib no-app no-makedep xrl xrl-include yrl yrl-include +CORE_APP_TARGETS = $(addprefix core-app-,$(CORE_APP_CASES)) +CORE_APP_CLEAN_TARGETS = $(addprefix clean-,$(CORE_APP_TARGETS)) + +.PHONY: core-app $(CORE_APP_TARGETS) clean-core-app $(CORE_APP_CLEAN_TARGETS) + +clean-core-app: $(CORE_APP_CLEAN_TARGETS) + +$(CORE_APP_CLEAN_TARGETS): + $t rm -rf $(APP_TO_CLEAN)/ + +core-app: $(CORE_APP_TARGETS) + +core-app-asn1: build clean-core-app-asn1 + + $i "Bootstrap a new OTP library named $(APP)" + $t mkdir $(APP)/ + $t cp ../erlang.mk $(APP)/ + $t $(MAKE) -C $(APP) -f erlang.mk bootstrap-lib $v + + $i "Download .asn1 files from Erlang/OTP" + $t mkdir $(APP)/asn1/ + $t curl -s -o $(APP)/asn1/CAP.asn1 $(OTP_MASTER)/lib/asn1/test/asn1_SUITE_data/CAP.asn1 + $t curl -s -o $(APP)/asn1/Def.asn1 $(OTP_MASTER)/lib/asn1/test/asn1_SUITE_data/Def.asn1 + + $i "Generate .erl files dependent from headers generated by .asn1 files" + $t printf "%s\n" "-module(use_cap)." "-include(\"CAP.hrl\")." > $(APP)/src/use_cap.erl + $t printf "%s\n" "-module(use_def)." "-include(\"Def.hrl\")." > $(APP)/src/use_def.erl + + $i "Generate an unrelated .hrl file" + $t mkdir $(APP)/include/ + $t touch $(APP)/include/unrelated.hrl + + $i "Build the application" + $t $(MAKE) -C $(APP) $v + + $i "Check that all compiled files exist" + $t test -f $(APP)/$(APP).d + $t test -f $(APP)/ebin/$(APP).app + $t test -f $(APP)/ebin/CAP.beam + $t test -f $(APP)/ebin/Def.beam + $t test -f $(APP)/ebin/use_cap.beam + $t test -f $(APP)/ebin/use_def.beam + $t test -f $(APP)/include/CAP.asn1db + $t test -f $(APP)/include/CAP.hrl + $t test -f $(APP)/include/Def.asn1db + $t test -f $(APP)/include/Def.hrl + $t test -f $(APP)/src/CAP.erl + $t test -f $(APP)/src/Def.erl + + $i "Check that the application was compiled correctly" + $t $(ERL) -pa $(APP)/ebin/ -eval " \ + ok = application:start($(APP)), \ + {ok, Mods = ['CAP', 'Def', use_cap, use_def]} \ + = application:get_key($(APP), modules), \ + [{module, M} = code:load_file(M) || M <- Mods], \ + halt()" + + $i "Touch one .asn1 file; check that only required files are rebuilt" +# The use_cap.erl gets touched because of its dependency to CAP.hrl. + $t printf "%s\n" \ + $(APP)/$(APP).d \ + $(APP)/ebin/$(APP).app \ + $(APP)/ebin/CAP.beam \ + $(APP)/ebin/use_cap.beam \ + $(APP)/include/CAP.asn1db \ + $(APP)/include/CAP.hrl \ + $(APP)/src/CAP.erl \ + $(APP)/src/use_cap.erl | sort > $(APP)/EXPECT + $t touch $(APP)/asn1/CAP.asn1 + $t $(MAKE) -C $(APP) $v + $t find $(APP) -type f -newer $(APP)/asn1/CAP.asn1 | sort | diff $(APP)/EXPECT - + $t rm $(APP)/EXPECT + + $i "Check that the application was compiled correctly" + $t $(ERL) -pa $(APP)/ebin/ -eval " \ + ok = application:start($(APP)), \ + {ok, Mods = ['CAP', 'Def', use_cap, use_def]} \ + = application:get_key($(APP), modules), \ + [{module, M} = code:load_file(M) || M <- Mods], \ + halt()" + + $i "Clean the application" + $t $(MAKE) -C $(APP) clean $v + + $i "Check that source files still exist" + $t test -f $(APP)/Makefile + $t test -f $(APP)/erlang.mk + $t test -f $(APP)/asn1/CAP.asn1 + $t test -f $(APP)/asn1/Def.asn1 + $t test -f $(APP)/include/unrelated.hrl +ifdef LEGACY + $t test -f $(APP)/src/$(APP).app.src +endif + $t test -f $(APP)/src/use_cap.erl + $t test -f $(APP)/src/use_def.erl + + $i "Check that all build artifacts are removed, including intermediates" + $t test ! -e $(APP)/$(APP).d + $t test ! -e $(APP)/ebin/ + $t test ! -e $(APP)/include/CAP.asn1db + $t test ! -e $(APP)/include/CAP.hrl + $t test ! -e $(APP)/include/Def.asn1db + $t test ! -e $(APP)/include/Def.hrl + $t test ! -e $(APP)/src/CAP.erl + $t test ! -e $(APP)/src/Def.erl + + $i "Build the application again" + $t $(MAKE) -C $(APP) $v + + $i "Check that all compiled files exist" + $t test -f $(APP)/$(APP).d + $t test -f $(APP)/ebin/$(APP).app + $t test -f $(APP)/ebin/CAP.beam + $t test -f $(APP)/ebin/Def.beam + $t test -f $(APP)/ebin/use_cap.beam + $t test -f $(APP)/ebin/use_def.beam + $t test -f $(APP)/include/CAP.asn1db + $t test -f $(APP)/include/CAP.hrl + $t test -f $(APP)/include/Def.asn1db + $t test -f $(APP)/include/Def.hrl + $t test -f $(APP)/src/CAP.erl + $t test -f $(APP)/src/Def.erl + + $i "Check that the application was compiled correctly" + $t $(ERL) -pa $(APP)/ebin/ -eval " \ + ok = application:start($(APP)), \ + {ok, Mods = ['CAP', 'Def', use_cap, use_def]} \ + = application:get_key($(APP), modules), \ + [{module, M} = code:load_file(M) || M <- Mods], \ + halt()" + +core-app-auto-git-id: build clean-core-app-auto-git-id + + $i "Bootstrap a new OTP library named $(APP)" + $t mkdir $(APP)/ + $t cp ../erlang.mk $(APP)/ + $t $(MAKE) -C $(APP) -f erlang.mk bootstrap-lib $v + + $i "Make it a git repository" + $t cd $(APP) && \ + git init -q && \ + git config user.email "[email protected]" && \ + git config user.name "test suite" && \ + git add . && \ + git commit -q -m "Tests" + + $i "Build the application" + $t $(MAKE) -C $(APP) $v + +ifdef LEGACY +# Legacy replaces {id, "git"} always regardless of built as a dependency. + $i "Check that the generated .app file has an id key" + $t $(ERL) -pa $(APP)/ebin/ -eval " \ + ok = application:start($(APP)), \ + {ok, ID} = application:get_key($(APP), id), \ + true = ID =/= [], \ + halt()" +else +# If there is no .app.src though, only fill in id when built as a dependency. + $i "Check that the generated .app file has no id key" + $t $(ERL) -pa $(APP)/ebin/ -eval " \ + ok = application:start($(APP)), \ + {ok, []} = application:get_key($(APP), id), \ + halt()" +endif + + $i "Clean the application" + $t $(MAKE) -C $(APP) clean $v + + $i "Build the application with IS_DEP=1" + $t $(MAKE) -C $(APP) IS_DEP=1 $v + + $i "Check that the generated .app file has an id key" + $t $(ERL) -pa $(APP)/ebin/ -eval " \ + ok = application:start($(APP)), \ + {ok, ID} = application:get_key($(APP), id), \ + true = ID =/= [], \ + halt()" + +core-app-erlc-exclude: build clean-core-app-erlc-exclude + + $i "Bootstrap a new OTP library named $(APP)" + $t mkdir $(APP)/ + $t cp ../erlang.mk $(APP)/ + $t $(MAKE) -C $(APP) -f erlang.mk bootstrap-lib $v + + $i "Generate .erl files" + $t echo "-module(boy)." > $(APP)/src/boy.erl + $t echo "-module(girl)." > $(APP)/src/girl.erl + + $i "Exclude boy.erl from the compilation" + $t echo "ERLC_EXCLUDE = boy" >> $(APP)/Makefile + + $i "Build the application" + $t $(MAKE) -C $(APP) $v + + $i "Check that boy.erl was not compiled" + $t test ! -e $(APP)/ebin/boy.beam + + $i "Check that the application was compiled correctly (without boy.erl)" + $t $(ERL) -pa $(APP)/ebin/ -eval " \ + ok = application:start($(APP)), \ + {ok, Mods = [girl]} \ + = application:get_key($(APP), modules), \ + [{module, M} = code:load_file(M) || M <- Mods], \ + halt()" + +core-app-erlc-opts: build clean-core-app-erlc-opts + + $i "Bootstrap a new OTP library named $(APP)" + $t mkdir $(APP)/ + $t cp ../erlang.mk $(APP)/ + $t $(MAKE) -C $(APP) -f erlang.mk bootstrap-lib $v + + $i "Define an empty ERLC_OPTS (without debug_info)" + $t echo "ERLC_OPTS =" >> $(APP)/Makefile + + $i "Generate .erl files" + $t echo "-module(boy)." > $(APP)/src/boy.erl + $t echo "-module(girl)." > $(APP)/src/girl.erl + + $i "Build the application" + $t $(MAKE) -C $(APP) $v + + $i "Check that the application was compiled correctly (without debug_info)" + $t $(ERL) -pa $(APP)/ebin/ -eval " \ + ok = application:start($(APP)), \ + {ok, Mods = [boy, girl]} \ + = application:get_key($(APP), modules), \ + [{module, M} = code:load_file(M) || M <- Mods], \ + false = proplists:is_defined(debug_info, proplists:get_value(options, boy:module_info(compile))), \ + false = proplists:is_defined(debug_info, proplists:get_value(options, girl:module_info(compile))), \ + halt()" + +core-app-erlc-opts-filter: build clean-core-app-erlc-opts-filter + + $i "Bootstrap a new OTP library named $(APP)" + $t mkdir $(APP)/ + $t cp ../erlang.mk $(APP)/ + $t $(MAKE) -C $(APP) -f erlang.mk bootstrap-lib $v + + $i "Define ERLC_OPTS filtering out debug_info" + $t echo "ERLC_OPTS := \$$(filter-out +debug_info,\$$(ERLC_OPTS))" >> $(APP)/Makefile + + $i "Generate .erl files" + $t echo "-module(boy)." > $(APP)/src/boy.erl + $t echo "-module(girl)." > $(APP)/src/girl.erl + + $i "Build the application" + $t $(MAKE) -C $(APP) $v + + $i "Check that the application was compiled correctly (without debug_info)" + $t $(ERL) -pa $(APP)/ebin/ -eval " \ + ok = application:start($(APP)), \ + {ok, Mods = [boy, girl]} \ + = application:get_key($(APP), modules), \ + [{module, M} = code:load_file(M) || M <- Mods], \ + false = proplists:is_defined(debug_info, proplists:get_value(options, boy:module_info(compile))), \ + false = proplists:is_defined(debug_info, proplists:get_value(options, girl:module_info(compile))), \ + halt()" + +core-app-error: build clean-core-app-error + + $i "Bootstrap a new OTP library named $(APP)" + $t mkdir $(APP)/ + $t cp ../erlang.mk $(APP)/ + $t $(MAKE) -C $(APP) -f erlang.mk bootstrap-lib $v + + $i "Generate a bad .erl files" + $t touch $(APP)/src/breaking.erl + + $i "Generate unrelated .erl files" + $t echo "-module(boy)." > $(APP)/src/boy.erl + $t echo "-module(girl)." > $(APP)/src/girl.erl + + $i "Check that trying to build returns non-zero" + $t if $(MAKE) -C $(APP) $v; then false; fi + +core-app-generate-erl: build clean-core-app-generate-erl + + $i "Bootstrap a new OTP library named $(APP)" + $t mkdir $(APP)/ + $t cp ../erlang.mk $(APP)/ + $t $(MAKE) -C $(APP) -f erlang.mk bootstrap-lib $v + + $i "Create a fake script file to be used as dependency" + $t touch $(APP)/script.sh + + $i "Append rules to the Makefile to generate a .erl module" + $t echo "\$$(PROJECT).d:: src/generated.erl" >> $(APP)/Makefile + $t echo "src/generated.erl:: script.sh; echo \"-module(generated).\" > \$$@" >> $(APP)/Makefile + + $i "Generate unrelated .erl files" + $t echo "-module(boy)." > $(APP)/src/boy.erl + $t echo "-module(girl)." > $(APP)/src/girl.erl + + $i "Build the application" + $t $(MAKE) -C $(APP) $v + + $i "Check that all compiled files exist" + $t test -f $(APP)/$(APP).d + $t test -f $(APP)/ebin/$(APP).app + $t test -f $(APP)/ebin/boy.beam + $t test -f $(APP)/ebin/generated.beam + $t test -f $(APP)/ebin/girl.beam + $t test -f $(APP)/src/generated.erl + + $i "Check that the application was compiled correctly" + $t $(ERL) -pa $(APP)/ebin/ -eval " \ + ok = application:start($(APP)), \ + {ok, Mods = [boy, generated, girl]} \ + = application:get_key($(APP), modules), \ + [{module, M} = code:load_file(M) || M <- Mods], \ + halt()" + + $i "Touch the script file; check that only required files are rebuilt" + $t printf "%s\n" \ + $(APP)/$(APP).d \ + $(APP)/ebin/$(APP).app \ + $(APP)/ebin/generated.beam \ + $(APP)/src/generated.erl | sort > $(APP)/EXPECT + $t touch $(APP)/script.sh + $t $(MAKE) -C $(APP) $v + $t find $(APP) -type f -newer $(APP)/script.sh | sort | diff $(APP)/EXPECT - + $t rm $(APP)/EXPECT + + $i "Check that the application was compiled correctly" + $t $(ERL) -pa $(APP)/ebin/ -eval " \ + ok = application:start($(APP)), \ + {ok, Mods = [boy, generated, girl]} \ + = application:get_key($(APP), modules), \ + [{module, M} = code:load_file(M) || M <- Mods], \ + halt()" + + $i "Clean the application" + $t $(MAKE) -C $(APP) clean $v + + $i "Check that source files still exist" + $t test -f $(APP)/Makefile + $t test -f $(APP)/erlang.mk + $t test -f $(APP)/script.sh +ifdef LEGACY + $t test -f $(APP)/src/$(APP).app.src +endif + $t test -f $(APP)/src/boy.erl + $t test -f $(APP)/src/girl.erl + + $i "Check that the generated .erl file still exists" + $t test -f $(APP)/src/generated.erl + + $i "Check that all build artifacts are removed" + $t test ! -e $(APP)/$(APP).d + $t test ! -e $(APP)/ebin/ + + $i "Add a rule to remove the generated .erl file on clean" + $t echo "clean:: ; rm src/generated.erl" >> $(APP)/Makefile + + $i "Clean the application again" + $t $(MAKE) -C $(APP) clean $v + + $i "Check that the generated .erl file was removed" + $t test ! -e $(APP)/src/generated.erl + + $i "Build the application again" + $t $(MAKE) -C $(APP) $v + + $i "Check that all compiled files exist" + $t test -f $(APP)/$(APP).d + $t test -f $(APP)/ebin/$(APP).app + $t test -f $(APP)/ebin/boy.beam + $t test -f $(APP)/ebin/generated.beam + $t test -f $(APP)/ebin/girl.beam + $t test -f $(APP)/src/generated.erl + + $i "Check that the application was compiled correctly" + $t $(ERL) -pa $(APP)/ebin/ -eval " \ + ok = application:start($(APP)), \ + {ok, Mods = [boy, generated, girl]} \ + = application:get_key($(APP), modules), \ + [{module, M} = code:load_file(M) || M <- Mods], \ + halt()" + +core-app-generate-erl-include: build clean-core-app-generate-erl-include + + $i "Bootstrap a new OTP library named $(APP)" + $t mkdir $(APP)/ + $t cp ../erlang.mk $(APP)/ + $t $(MAKE) -C $(APP) -f erlang.mk bootstrap-lib $v + + $i "Create a fake script file to be used as dependency" + $t touch $(APP)/script.sh + + $i "Append rules to the Makefile to generate a .erl module" + $t echo "\$$(PROJECT).d:: src/generated.erl" >> $(APP)/Makefile + $t echo "src/generated.erl:: script.sh; echo \"-module(generated).\" > \$$@; echo \"-include(\\\"included.hrl\\\").\" >> \$$@" >> $(APP)/Makefile + + $i "Generate the .hrl file" + $t mkdir $(APP)/include/ + $t touch $(APP)/include/included.hrl + + $i "Generate unrelated .erl files" + $t echo "-module(boy)." > $(APP)/src/boy.erl + $t echo "-module(girl)." > $(APP)/src/girl.erl + + $i "Build the application" + $t $(MAKE) -C $(APP) $v + + $i "Check that all compiled files exist" + $t test -f $(APP)/$(APP).d + $t test -f $(APP)/ebin/$(APP).app + $t test -f $(APP)/ebin/boy.beam + $t test -f $(APP)/ebin/generated.beam + $t test -f $(APP)/ebin/girl.beam + $t test -f $(APP)/src/generated.erl + + $i "Check that the application was compiled correctly" + $t $(ERL) -pa $(APP)/ebin/ -eval " \ + ok = application:start($(APP)), \ + {ok, Mods = [boy, generated, girl]} \ + = application:get_key($(APP), modules), \ + [{module, M} = code:load_file(M) || M <- Mods], \ + halt()" + + $i "Touch the .hrl file; check that only required files are rebuilt" + $t printf "%s\n" \ + $(APP)/$(APP).d \ + $(APP)/ebin/$(APP).app \ + $(APP)/ebin/generated.beam \ + $(APP)/src/generated.erl | sort > $(APP)/EXPECT + $t touch $(APP)/include/included.hrl + $t $(MAKE) -C $(APP) $v + $t find $(APP) -type f -newer $(APP)/include/included.hrl | sort | diff $(APP)/EXPECT - + $t rm $(APP)/EXPECT + + $i "Check that the application was compiled correctly" + $t $(ERL) -pa $(APP)/ebin/ -eval " \ + ok = application:start($(APP)), \ + {ok, Mods = [boy, generated, girl]} \ + = application:get_key($(APP), modules), \ + [{module, M} = code:load_file(M) || M <- Mods], \ + halt()" + +core-app-generate-erl-prepend: build clean-core-app-generate-erl-prepend + + $i "Bootstrap a new OTP library named $(APP)" + $t mkdir $(APP)/ + $t cp ../erlang.mk $(APP)/ + $t $(MAKE) -C $(APP) -f erlang.mk bootstrap-lib $v + + $i "Create a fake script file to be used as dependency" + $t touch $(APP)/script.sh + + $i "Generate a Makefile and prepend rules that generate a .erl module" + $t echo "PROJECT = $(APP)" > $(APP)/Makefile + $t echo ".DEFAULT_GOAL = all" >> $(APP)/Makefile + $t echo "\$$(PROJECT).d:: src/generated.erl" >> $(APP)/Makefile + $t echo "src/generated.erl:: script.sh; echo \"-module(generated).\" > \$$@" >> $(APP)/Makefile + $t echo "include erlang.mk" >> $(APP)/Makefile + + $i "Generate unrelated .erl files" + $t echo "-module(boy)." > $(APP)/src/boy.erl + $t echo "-module(girl)." > $(APP)/src/girl.erl + + $i "Build the application" + $t $(MAKE) -C $(APP) $v + + $i "Check that all compiled files exist" + $t test -f $(APP)/$(APP).d + $t test -f $(APP)/ebin/$(APP).app + $t test -f $(APP)/ebin/boy.beam + $t test -f $(APP)/ebin/generated.beam + $t test -f $(APP)/ebin/girl.beam + $t test -f $(APP)/src/generated.erl + + $i "Check that the application was compiled correctly" + $t $(ERL) -pa $(APP)/ebin/ -eval " \ + ok = application:start($(APP)), \ + {ok, Mods = [boy, generated, girl]} \ + = application:get_key($(APP), modules), \ + [{module, M} = code:load_file(M) || M <- Mods], \ + halt()" + + $i "Touch the script file; check that only required files are rebuilt" + $t printf "%s\n" \ + $(APP)/$(APP).d \ + $(APP)/ebin/$(APP).app \ + $(APP)/ebin/generated.beam \ + $(APP)/src/generated.erl | sort > $(APP)/EXPECT + $t touch $(APP)/script.sh + $t $(MAKE) -C $(APP) $v + $t find $(APP) -type f -newer $(APP)/script.sh | sort | diff $(APP)/EXPECT - + $t rm $(APP)/EXPECT + + $i "Check that the application was compiled correctly" + $t $(ERL) -pa $(APP)/ebin/ -eval " \ + ok = application:start($(APP)), \ + {ok, Mods = [boy, generated, girl]} \ + = application:get_key($(APP), modules), \ + [{module, M} = code:load_file(M) || M <- Mods], \ + halt()" + + $i "Clean the application" + $t $(MAKE) -C $(APP) clean $v + + $i "Check that source files still exist" + $t test -f $(APP)/Makefile + $t test -f $(APP)/erlang.mk + $t test -f $(APP)/script.sh +ifdef LEGACY + $t test -f $(APP)/src/$(APP).app.src +endif + $t test -f $(APP)/src/boy.erl + $t test -f $(APP)/src/girl.erl + + $i "Check that the generated .erl file still exists" + $t test -f $(APP)/src/generated.erl + + $i "Check that all build artifacts are removed" + $t test ! -e $(APP)/$(APP).d + $t test ! -e $(APP)/ebin/ + + $i "Add a rule to remove the generated .erl file on clean" + $t echo "clean:: ; rm src/generated.erl" >> $(APP)/Makefile + + $i "Clean the application again" + $t $(MAKE) -C $(APP) clean $v + + $i "Check that the generated .erl file was removed" + $t test ! -e $(APP)/src/generated.erl + + $i "Build the application again" + $t $(MAKE) -C $(APP) $v + + $i "Check that all compiled files exist" + $t test -f $(APP)/$(APP).d + $t test -f $(APP)/ebin/$(APP).app + $t test -f $(APP)/ebin/boy.beam + $t test -f $(APP)/ebin/generated.beam + $t test -f $(APP)/ebin/girl.beam + $t test -f $(APP)/src/generated.erl + + $i "Check that the application was compiled correctly" + $t $(ERL) -pa $(APP)/ebin/ -eval " \ + ok = application:start($(APP)), \ + {ok, Mods = [boy, generated, girl]} \ + = application:get_key($(APP), modules), \ + [{module, M} = code:load_file(M) || M <- Mods], \ + halt()" + +core-app-hrl: build clean-core-app-hrl + + $i "Bootstrap a new OTP library named $(APP)" + $t mkdir $(APP)/ + $t cp ../erlang.mk $(APP)/ + $t $(MAKE) -C $(APP) -f erlang.mk bootstrap-lib $v + + $i "Generate .hrl files" + $t mkdir $(APP)/include/ + $t touch $(APP)/include/blue.hrl $(APP)/include/red.hrl + + $i "Generate .erl files dependent from headers" + $t printf "%s\n" "-module(use_blue)." "-include(\"blue.hrl\")." > $(APP)/src/use_blue.erl + $t printf "%s\n" "-module(use_red)." "-include(\"red.hrl\")." > $(APP)/src/use_red.erl + + $i "Build the application" + $t $(MAKE) -C $(APP) $v + + $i "Check that all compiled files exist" + $t test -f $(APP)/$(APP).d + $t test -f $(APP)/ebin/$(APP).app + $t test -f $(APP)/ebin/use_blue.beam + $t test -f $(APP)/ebin/use_red.beam + + $i "Check that the application was compiled correctly" + $t $(ERL) -pa $(APP)/ebin/ -eval " \ + ok = application:start($(APP)), \ + {ok, Mods = [use_blue, use_red]} \ + = application:get_key($(APP), modules), \ + [{module, M} = code:load_file(M) || M <- Mods], \ + halt()" + + $i "Touch one .hrl file; check that only required files are rebuilt" +# The use_red.erl gets touched because of its dependency to red.hrl. + $t printf "%s\n" \ + $(APP)/$(APP).d \ + $(APP)/ebin/$(APP).app \ + $(APP)/ebin/use_red.beam \ + $(APP)/src/use_red.erl | sort > $(APP)/EXPECT + $t touch $(APP)/include/red.hrl + $t $(MAKE) -C $(APP) $v + $t find $(APP) -type f -newer $(APP)/include/red.hrl | sort | diff $(APP)/EXPECT - + $t rm $(APP)/EXPECT + + $i "Check that the application was compiled correctly" + $t $(ERL) -pa $(APP)/ebin/ -eval " \ + ok = application:start($(APP)), \ + {ok, Mods = [use_blue, use_red]} \ + = application:get_key($(APP), modules), \ + [{module, M} = code:load_file(M) || M <- Mods], \ + halt()" + + $i "Clean the application" + $t $(MAKE) -C $(APP) clean $v + + $i "Check that source files still exist" + $t test -f $(APP)/Makefile + $t test -f $(APP)/erlang.mk + $t test -f $(APP)/include/blue.hrl + $t test -f $(APP)/include/red.hrl +ifdef LEGACY + $t test -f $(APP)/src/$(APP).app.src +endif + $t test -f $(APP)/src/use_blue.erl + $t test -f $(APP)/src/use_red.erl + + $i "Check that all build artifacts are removed" + $t test ! -e $(APP)/$(APP).d + $t test ! -e $(APP)/ebin/ + + $i "Build the application again" + $t $(MAKE) -C $(APP) $v + + $i "Check that all compiled files exist" + $t test -f $(APP)/$(APP).d + $t test -f $(APP)/ebin/$(APP).app + $t test -f $(APP)/ebin/use_blue.beam + $t test -f $(APP)/ebin/use_red.beam + + $i "Check that the application was compiled correctly" + $t $(ERL) -pa $(APP)/ebin/ -eval " \ + ok = application:start($(APP)), \ + {ok, Mods = [use_blue, use_red]} \ + = application:get_key($(APP), modules), \ + [{module, M} = code:load_file(M) || M <- Mods], \ + halt()" + +core-app-hrl-recursive: build clean-core-app-hrl-recursive + + $i "Bootstrap a new OTP library named $(APP)" + $t mkdir $(APP)/ + $t cp ../erlang.mk $(APP)/ + $t $(MAKE) -C $(APP) -f erlang.mk bootstrap-lib $v + + $i "Generate .hrl files" + $t mkdir $(APP)/include/ + $t touch $(APP)/include/blue.hrl $(APP)/include/pill.hrl + $t echo "-include(\"pill.hrl\")." > $(APP)/include/red.hrl + + $i "Generate .erl files dependent from headers" + $t printf "%s\n" "-module(use_blue)." "-include(\"blue.hrl\")." > $(APP)/src/use_blue.erl + $t printf "%s\n" "-module(use_red)." "-include(\"red.hrl\")." > $(APP)/src/use_red.erl + + $i "Build the application" + $t $(MAKE) -C $(APP) $v + + $i "Check that all compiled files exist" + $t test -f $(APP)/$(APP).d + $t test -f $(APP)/ebin/$(APP).app + $t test -f $(APP)/ebin/use_blue.beam + $t test -f $(APP)/ebin/use_red.beam + + $i "Check that the application was compiled correctly" + $t $(ERL) -pa $(APP)/ebin/ -eval " \ + ok = application:start($(APP)), \ + {ok, Mods = [use_blue, use_red]} \ + = application:get_key($(APP), modules), \ + [{module, M} = code:load_file(M) || M <- Mods], \ + halt()" + + $i "Touch the deepest .hrl file; check that only required files are rebuilt" +# The use_red.erl gets touched because of its dependency to red.hrl and pill.hrl. + $t printf "%s\n" \ + $(APP)/$(APP).d \ + $(APP)/ebin/$(APP).app \ + $(APP)/ebin/use_red.beam \ + $(APP)/src/use_red.erl | sort > $(APP)/EXPECT + $t touch $(APP)/include/pill.hrl + $t $(MAKE) -C $(APP) $v + $t find $(APP) -type f -newer $(APP)/include/pill.hrl | sort | diff $(APP)/EXPECT - + $t rm $(APP)/EXPECT + + $i "Check that the application was compiled correctly" + $t $(ERL) -pa $(APP)/ebin/ -eval " \ + ok = application:start($(APP)), \ + {ok, Mods = [use_blue, use_red]} \ + = application:get_key($(APP), modules), \ + [{module, M} = code:load_file(M) || M <- Mods], \ + halt()" + + $i "Clean the application" + $t $(MAKE) -C $(APP) clean $v + + $i "Check that source files still exist" + $t test -f $(APP)/Makefile + $t test -f $(APP)/erlang.mk + $t test -f $(APP)/include/blue.hrl + $t test -f $(APP)/include/pill.hrl + $t test -f $(APP)/include/red.hrl +ifdef LEGACY + $t test -f $(APP)/src/$(APP).app.src +endif + $t test -f $(APP)/src/use_blue.erl + $t test -f $(APP)/src/use_red.erl + + $i "Check that all build artifacts are removed" + $t test ! -e $(APP)/$(APP).d + $t test ! -e $(APP)/ebin/ + + $i "Build the application again" + $t $(MAKE) -C $(APP) $v + + $i "Check that all compiled files exist" + $t test -f $(APP)/$(APP).d + $t test -f $(APP)/ebin/$(APP).app + $t test -f $(APP)/ebin/use_blue.beam + $t test -f $(APP)/ebin/use_red.beam + + $i "Check that the application was compiled correctly" + $t $(ERL) -pa $(APP)/ebin/ -eval " \ + ok = application:start($(APP)), \ + {ok, Mods = [use_blue, use_red]} \ + = application:get_key($(APP), modules), \ + [{module, M} = code:load_file(M) || M <- Mods], \ + halt()" + +core-app-mib: build clean-core-app-mib + + $i "Bootstrap a new OTP library named $(APP)" + $t mkdir $(APP)/ + $t cp ../erlang.mk $(APP)/ + $t $(MAKE) -C $(APP) -f erlang.mk bootstrap-lib $v + + $i "Download .mib files from Erlang/OTP" + $t mkdir $(APP)/mibs/ + $t curl -s -o $(APP)/mibs/EX1-MIB.mib $(OTP_MASTER)/lib/snmp/examples/ex1/EX1-MIB.mib + $t curl -s -o $(APP)/mibs/OTP-REG.mib $(OTP_MASTER)/lib/otp_mibs/mibs/OTP-REG.mib + + $i "Generate .erl files dependent from headers generated by .mib files" + $t printf "%s\n" "-module(use_v1)." "-include(\"EX1-MIB.hrl\")." > $(APP)/src/use_v1.erl + $t printf "%s\n" "-module(use_v2)." "-include(\"OTP-REG.hrl\")." > $(APP)/src/use_v2.erl + + $i "Generate an unrelated .hrl file" + $t mkdir $(APP)/include/ + $t touch $(APP)/include/unrelated.hrl + + $i "Build the application" + $t $(MAKE) -C $(APP) $v + + $i "Check that all compiled files exist" + $t test -f $(APP)/$(APP).d + $t test -f $(APP)/ebin/$(APP).app + $t test -f $(APP)/ebin/use_v1.beam + $t test -f $(APP)/ebin/use_v2.beam + $t test -f $(APP)/include/EX1-MIB.hrl + $t test -f $(APP)/include/OTP-REG.hrl + $t test -f $(APP)/priv/mibs/EX1-MIB.bin + $t test -f $(APP)/priv/mibs/OTP-REG.bin + + $i "Check that the application was compiled correctly" + $t $(ERL) -pa $(APP)/ebin/ -eval " \ + ok = application:start($(APP)), \ + {ok, Mods = [use_v1, use_v2]} \ + = application:get_key($(APP), modules), \ + [{module, M} = code:load_file(M) || M <- Mods], \ + halt()" + + $i "Touch one .mib file; check that only required files are rebuilt" +# The use_v1.erl gets touched because of its dependency to EX1-MIB.hrl. + $t printf "%s\n" \ + $(APP)/$(APP).d \ + $(APP)/ebin/$(APP).app \ + $(APP)/ebin/use_v1.beam \ + $(APP)/include/EX1-MIB.hrl \ + $(APP)/priv/mibs/EX1-MIB.bin \ + $(APP)/src/use_v1.erl | sort > $(APP)/EXPECT + $t touch $(APP)/mibs/EX1-MIB.mib + $t $(MAKE) -C $(APP) $v + $t find $(APP) -type f -newer $(APP)/mibs/EX1-MIB.mib | sort | diff $(APP)/EXPECT - + $t rm $(APP)/EXPECT + + $i "Check that the application was compiled correctly" + $t $(ERL) -pa $(APP)/ebin/ -eval " \ + ok = application:start($(APP)), \ + {ok, Mods = [use_v1, use_v2]} \ + = application:get_key($(APP), modules), \ + [{module, M} = code:load_file(M) || M <- Mods], \ + halt()" + + $i "Clean the application" + $t $(MAKE) -C $(APP) clean $v + + $i "Check that source files still exist" + $t test -f $(APP)/Makefile + $t test -f $(APP)/erlang.mk + $t test -f $(APP)/include/unrelated.hrl + $t test -f $(APP)/mibs/EX1-MIB.mib + $t test -f $(APP)/mibs/OTP-REG.mib +ifdef LEGACY + $t test -f $(APP)/src/$(APP).app.src +endif + $t test -f $(APP)/src/use_v1.erl + $t test -f $(APP)/src/use_v2.erl + + $i "Check that all build artifacts are removed, including intermediates" + $t test ! -e $(APP)/$(APP).d + $t test ! -e $(APP)/ebin/ + $t test ! -e $(APP)/include/EX1-MIB.hrl + $t test ! -e $(APP)/include/OTP-REG.hrl + $t test ! -e $(APP)/priv/mibs/ + + $i "Build the application again" + $t $(MAKE) -C $(APP) $v + + $i "Check that all compiled files exist" + $t test -f $(APP)/$(APP).d + $t test -f $(APP)/ebin/$(APP).app + $t test -f $(APP)/ebin/use_v1.beam + $t test -f $(APP)/ebin/use_v2.beam + $t test -f $(APP)/include/EX1-MIB.hrl + $t test -f $(APP)/include/OTP-REG.hrl + $t test -f $(APP)/priv/mibs/EX1-MIB.bin + $t test -f $(APP)/priv/mibs/OTP-REG.bin + + $i "Check that the application was compiled correctly" + $t $(ERL) -pa $(APP)/ebin/ -eval " \ + ok = application:start($(APP)), \ + {ok, Mods = [use_v1, use_v2]} \ + = application:get_key($(APP), modules), \ + [{module, M} = code:load_file(M) || M <- Mods], \ + halt()" + +core-app-no-app: build clean-core-app-no-app + + $i "Bootstrap a project without an OTP library" + $t mkdir $(APP)/ + $t cp ../erlang.mk $(APP)/ + $t $(MAKE) -C $(APP) -f erlang.mk bootstrap-lib $v + $t rm -rf $(APP)/src + + $i "Build the project" + $t $(MAKE) -C $(APP) $v + +core-app-no-makedep: build clean-core-app-no-makedep + + $i "Bootstrap a new OTP library named $(APP)" + $t mkdir $(APP)/ + $t cp ../erlang.mk $(APP)/ + $t $(MAKE) -C $(APP) -f erlang.mk bootstrap-lib $v + + $i "Set NO_MAKEDEP ?= 1 in the Makefile" + $t sed -i.bak '2i\ +NO_MAKEDEP ?= 1\ +' $(APP)/Makefile + + $i "Generate .hrl files" + $t mkdir $(APP)/include/ + $t touch $(APP)/include/blue.hrl $(APP)/include/red.hrl + + $i "Generate .erl files dependent from headers" + $t printf "%s\n" "-module(use_blue)." "-include(\"blue.hrl\")." > $(APP)/src/use_blue.erl + $t printf "%s\n" "-module(use_red)." "-include(\"red.hrl\")." > $(APP)/src/use_red.erl + + $i "Build the application" + $t $(MAKE) -C $(APP) $v + + $i "Check that all compiled files exist" + $t test -f $(APP)/$(APP).d + $t test -f $(APP)/ebin/$(APP).app + $t test -f $(APP)/ebin/use_blue.beam + $t test -f $(APP)/ebin/use_red.beam + + $i "Check that the application was compiled correctly" + $t $(ERL) -pa $(APP)/ebin/ -eval " \ + ok = application:start($(APP)), \ + {ok, Mods = [use_blue, use_red]} \ + = application:get_key($(APP), modules), \ + [{module, M} = code:load_file(M) || M <- Mods], \ + halt()" + + $i "Touch one .hrl file; check that only required files are rebuilt" +# The use_red.erl gets touched because of its dependency to red.hrl. + $t printf "%s\n" \ + $(APP)/ebin/$(APP).app \ + $(APP)/ebin/use_red.beam \ + $(APP)/src/use_red.erl | sort > $(APP)/EXPECT + $t touch $(APP)/include/red.hrl + $t $(MAKE) -C $(APP) $v + $t find $(APP) -type f -newer $(APP)/include/red.hrl | sort | diff $(APP)/EXPECT - + $t rm $(APP)/EXPECT + + $i "Check that the application was compiled correctly" + $t $(ERL) -pa $(APP)/ebin/ -eval " \ + ok = application:start($(APP)), \ + {ok, Mods = [use_blue, use_red]} \ + = application:get_key($(APP), modules), \ + [{module, M} = code:load_file(M) || M <- Mods], \ + halt()" + + $i "Touch one .hrl file; disable NO_MAKEDEP and check that only required files are rebuilt" +# The use_red.erl gets touched because of its dependency to red.hrl. + $t printf "%s\n" \ + $(APP)/$(APP).d \ + $(APP)/ebin/$(APP).app \ + $(APP)/ebin/use_red.beam \ + $(APP)/src/use_red.erl | sort > $(APP)/EXPECT + $t touch $(APP)/include/red.hrl + $t NO_MAKEDEP= $(MAKE) -C $(APP) $v + $t find $(APP) -type f -newer $(APP)/include/red.hrl | sort | diff $(APP)/EXPECT - + $t rm $(APP)/EXPECT + + $i "Check that the application was compiled correctly" + $t $(ERL) -pa $(APP)/ebin/ -eval " \ + ok = application:start($(APP)), \ + {ok, Mods = [use_blue, use_red]} \ + = application:get_key($(APP), modules), \ + [{module, M} = code:load_file(M) || M <- Mods], \ + halt()" + + $i "Clean the application" + $t $(MAKE) -C $(APP) clean $v + + $i "Check that source files still exist" + $t test -f $(APP)/Makefile + $t test -f $(APP)/erlang.mk + $t test -f $(APP)/include/blue.hrl + $t test -f $(APP)/include/red.hrl +ifdef LEGACY + $t test -f $(APP)/src/$(APP).app.src +endif + $t test -f $(APP)/src/use_blue.erl + $t test -f $(APP)/src/use_red.erl + + $i "Check that all build artifacts are removed" + $t test ! -e $(APP)/$(APP).d + $t test ! -e $(APP)/ebin/ + + $i "Build the application again" + $t $(MAKE) -C $(APP) $v + + $i "Check that all compiled files exist" + $t test -f $(APP)/$(APP).d + $t test -f $(APP)/ebin/$(APP).app + $t test -f $(APP)/ebin/use_blue.beam + $t test -f $(APP)/ebin/use_red.beam + + $i "Check that the application was compiled correctly" + $t $(ERL) -pa $(APP)/ebin/ -eval " \ + ok = application:start($(APP)), \ + {ok, Mods = [use_blue, use_red]} \ + = application:get_key($(APP), modules), \ + [{module, M} = code:load_file(M) || M <- Mods], \ + halt()" + +core-app-xrl: build clean-core-app-xrl + + $i "Bootstrap a new OTP library named $(APP)" + $t mkdir $(APP)/ + $t cp ../erlang.mk $(APP)/ + $t $(MAKE) -C $(APP) -f erlang.mk bootstrap-lib $v + + $i "Download .xrl files from Robert" + $t curl -s -o $(APP)/src/erlang_scan.xrl https://raw.githubusercontent.com/rvirding/leex/master/examples/erlang_scan.xrl + $t curl -s -o $(APP)/src/lfe_scan.xrl https://raw.githubusercontent.com/rvirding/leex/master/examples/lfe_scan.xrl + + $i "Generate unrelated .erl files" + $t echo "-module(boy)." > $(APP)/src/boy.erl + $t echo "-module(girl)." > $(APP)/src/girl.erl + + $i "Disable warnings; our test .xrl files aren't perfect" + $t echo "ERLC_OPTS=+debug_info" >> $(APP)/Makefile + + $i "Build the application" + $t $(MAKE) -C $(APP) $v + + $i "Check that all compiled files exist" + $t test -f $(APP)/$(APP).d + $t test -f $(APP)/ebin/$(APP).app + $t test -f $(APP)/ebin/boy.beam + $t test -f $(APP)/ebin/erlang_scan.beam + $t test -f $(APP)/ebin/girl.beam + $t test -f $(APP)/ebin/lfe_scan.beam + $t test -f $(APP)/src/erlang_scan.erl + $t test -f $(APP)/src/lfe_scan.erl + + $i "Check that the application was compiled correctly" + $t $(ERL) -pa $(APP)/ebin/ -eval " \ + ok = application:start($(APP)), \ + {ok, Mods = [boy, erlang_scan, girl, lfe_scan]} \ + = application:get_key($(APP), modules), \ + [{module, M} = code:load_file(M) || M <- Mods], \ + halt()" + + $i "Touch one .xrl file; check that only required files are rebuilt" + $t printf "%s\n" \ + $(APP)/$(APP).d \ + $(APP)/ebin/$(APP).app \ + $(APP)/ebin/erlang_scan.beam \ + $(APP)/src/erlang_scan.erl | sort > $(APP)/EXPECT + $t touch $(APP)/src/erlang_scan.xrl + $t $(MAKE) -C $(APP) $v + $t find $(APP) -type f -newer $(APP)/src/erlang_scan.xrl | sort | diff $(APP)/EXPECT - + $t rm $(APP)/EXPECT + + $i "Check that the application was compiled correctly" + $t $(ERL) -pa $(APP)/ebin/ -eval " \ + ok = application:start($(APP)), \ + {ok, Mods = [boy, erlang_scan, girl, lfe_scan]} \ + = application:get_key($(APP), modules), \ + [{module, M} = code:load_file(M) || M <- Mods], \ + halt()" + + $i "Clean the application" + $t $(MAKE) -C $(APP) clean $v + + $i "Check that source files still exist" + $t test -f $(APP)/Makefile + $t test -f $(APP)/erlang.mk +ifdef LEGACY + $t test -f $(APP)/src/$(APP).app.src +endif + $t test -f $(APP)/src/boy.erl + $t test -f $(APP)/src/erlang_scan.xrl + $t test -f $(APP)/src/girl.erl + $t test -f $(APP)/src/lfe_scan.xrl + + $i "Check that all build artifacts are removed, including intermediates" + $t test ! -e $(APP)/$(APP).d + $t test ! -e $(APP)/ebin/ + $t test ! -e $(APP)/src/erlang_scan.erl + $t test ! -e $(APP)/src/lfe_scan.erl + + $i "Build the application again" + $t $(MAKE) -C $(APP) $v + + $i "Check that all compiled files exist" + $t test -f $(APP)/$(APP).d + $t test -f $(APP)/ebin/$(APP).app + $t test -f $(APP)/ebin/boy.beam + $t test -f $(APP)/ebin/erlang_scan.beam + $t test -f $(APP)/ebin/girl.beam + $t test -f $(APP)/ebin/lfe_scan.beam + $t test -f $(APP)/src/erlang_scan.erl + $t test -f $(APP)/src/lfe_scan.erl + + $i "Check that the application was compiled correctly" + $t $(ERL) -pa $(APP)/ebin/ -eval " \ + ok = application:start($(APP)), \ + {ok, Mods = [boy, erlang_scan, girl, lfe_scan]} \ + = application:get_key($(APP), modules), \ + [{module, M} = code:load_file(M) || M <- Mods], \ + halt()" + +core-app-xrl-include: build clean-core-app-xrl-include + + $i "Bootstrap a new OTP library named $(APP)" + $t mkdir $(APP)/ + $t cp ../erlang.mk $(APP)/ + $t $(MAKE) -C $(APP) -f erlang.mk bootstrap-lib $v + + $i "Download a .xrl file with numerous includes from Gordon" + $t curl -s -o $(APP)/src/xfl_lexer.xrl https://raw.githubusercontent.com/hypernumbers/hypernumbers/master/lib/formula_engine-1.0/priv/xfl_lexer.xrl + $t curl -s -o $(APP)/src/errvals.hrl https://raw.githubusercontent.com/hypernumbers/hypernumbers/master/lib/hypernumbers-1.0/include/errvals.hrl + $t curl -s -o $(APP)/src/muin_proc_dict.hrl https://raw.githubusercontent.com/hypernumbers/hypernumbers/master/lib/hypernumbers-1.0/include/muin_proc_dict.hrl + $t curl -s -o $(APP)/src/muin_records.hrl https://raw.githubusercontent.com/hypernumbers/hypernumbers/master/lib/hypernumbers-1.0/include/muin_records.hrl + $t curl -s -o $(APP)/src/typechecks.hrl https://raw.githubusercontent.com/hypernumbers/hypernumbers/master/lib/hypernumbers-1.0/include/typechecks.hrl + + $i "Generate unrelated .erl files" + $t echo "-module(boy)." > $(APP)/src/boy.erl + $t echo "-module(girl)." > $(APP)/src/girl.erl + + $i "Disable warnings; our test .xrl files aren't perfect" + $t echo "ERLC_OPTS=+debug_info" >> $(APP)/Makefile + + $i "Build the application" + $t $(MAKE) -C $(APP) $v + + $i "Check that all compiled files exist" + $t test -f $(APP)/$(APP).d + $t test -f $(APP)/ebin/$(APP).app + $t test -f $(APP)/ebin/boy.beam + $t test -f $(APP)/ebin/girl.beam + $t test -f $(APP)/ebin/xfl_lexer.beam + $t test -f $(APP)/src/xfl_lexer.erl + + $i "Check that the application was compiled correctly" + $t $(ERL) -pa $(APP)/ebin/ -eval " \ + ok = application:start($(APP)), \ + {ok, Mods = [boy, girl, xfl_lexer]} \ + = application:get_key($(APP), modules), \ + [{module, M} = code:load_file(M) || M <- Mods], \ + halt()" + + $i "Touch the .xrl file; check that only required files are rebuilt" + $t printf "%s\n" \ + $(APP)/$(APP).d \ + $(APP)/ebin/$(APP).app \ + $(APP)/ebin/xfl_lexer.beam \ + $(APP)/src/xfl_lexer.erl | sort > $(APP)/EXPECT + $t touch $(APP)/src/xfl_lexer.xrl + $t $(MAKE) -C $(APP) $v + $t find $(APP) -type f -newer $(APP)/src/xfl_lexer.xrl | sort | diff $(APP)/EXPECT - + $t rm $(APP)/EXPECT + + $i "Check that the application was compiled correctly" + $t $(ERL) -pa $(APP)/ebin/ -eval " \ + ok = application:start($(APP)), \ + {ok, Mods = [boy, girl, xfl_lexer]} \ + = application:get_key($(APP), modules), \ + [{module, M} = code:load_file(M) || M <- Mods], \ + halt()" + + $i "Touch a .hrl file included directly; check that only required files are rebuilt" + $t printf "%s\n" \ + $(APP)/$(APP).d \ + $(APP)/ebin/$(APP).app \ + $(APP)/ebin/xfl_lexer.beam \ + $(APP)/src/xfl_lexer.erl | sort > $(APP)/EXPECT + $t touch $(APP)/src/typechecks.hrl + $t $(MAKE) -C $(APP) $v + $t find $(APP) -type f -newer $(APP)/src/typechecks.hrl | sort | diff $(APP)/EXPECT - + $t rm $(APP)/EXPECT + + $i "Check that the application was compiled correctly" + $t $(ERL) -pa $(APP)/ebin/ -eval " \ + ok = application:start($(APP)), \ + {ok, Mods = [boy, girl, xfl_lexer]} \ + = application:get_key($(APP), modules), \ + [{module, M} = code:load_file(M) || M <- Mods], \ + halt()" + + $i "Touch a .hrl file included indirectly; check that only required files are rebuilt" + $t printf "%s\n" \ + $(APP)/$(APP).d \ + $(APP)/ebin/$(APP).app \ + $(APP)/ebin/xfl_lexer.beam \ + $(APP)/src/xfl_lexer.erl | sort > $(APP)/EXPECT + $t touch $(APP)/src/errvals.hrl + $t $(MAKE) -C $(APP) $v + $t find $(APP) -type f -newer $(APP)/src/errvals.hrl | sort | diff $(APP)/EXPECT - + $t rm $(APP)/EXPECT + + $i "Check that the application was compiled correctly" + $t $(ERL) -pa $(APP)/ebin/ -eval " \ + ok = application:start($(APP)), \ + {ok, Mods = [boy, girl, xfl_lexer]} \ + = application:get_key($(APP), modules), \ + [{module, M} = code:load_file(M) || M <- Mods], \ + halt()" + + $i "Clean the application" + $t $(MAKE) -C $(APP) clean $v + + $i "Check that source files still exist" + $t test -f $(APP)/Makefile + $t test -f $(APP)/erlang.mk +ifdef LEGACY + $t test -f $(APP)/src/$(APP).app.src +endif + $t test -f $(APP)/src/boy.erl + $t test -f $(APP)/src/girl.erl + $t test -f $(APP)/src/errvals.hrl + $t test -f $(APP)/src/muin_proc_dict.hrl + $t test -f $(APP)/src/muin_records.hrl + $t test -f $(APP)/src/typechecks.hrl + $t test -f $(APP)/src/xfl_lexer.xrl + + $i "Check that all build artifacts are removed, including intermediates" + $t test ! -e $(APP)/$(APP).d + $t test ! -e $(APP)/ebin/ + $t test ! -e $(APP)/src/xfl_lexer.erl + + $i "Build the application again" + $t $(MAKE) -C $(APP) $v + + $i "Check that all compiled files exist" + $t test -f $(APP)/$(APP).d + $t test -f $(APP)/ebin/$(APP).app + $t test -f $(APP)/ebin/boy.beam + $t test -f $(APP)/ebin/girl.beam + $t test -f $(APP)/ebin/xfl_lexer.beam + $t test -f $(APP)/src/xfl_lexer.erl + + $i "Check that the application was compiled correctly" + $t $(ERL) -pa $(APP)/ebin/ -eval " \ + ok = application:start($(APP)), \ + {ok, Mods = [boy, girl, xfl_lexer]} \ + = application:get_key($(APP), modules), \ + [{module, M} = code:load_file(M) || M <- Mods], \ + halt()" + +core-app-yrl: build clean-core-app-yrl + + $i "Bootstrap a new OTP library named $(APP)" + $t mkdir $(APP)/ + $t cp ../erlang.mk $(APP)/ + $t $(MAKE) -C $(APP) -f erlang.mk bootstrap-lib $v + + $i "Download .yrl files from Erlang/OTP" + $t curl -s -o $(APP)/src/xmerl_xpath_parse.yrl $(OTP_MASTER)/lib/xmerl/src/xmerl_xpath_parse.yrl + $t curl -s -o $(APP)/src/xref_parser.yrl $(OTP_MASTER)/lib/tools/src/xref_parser.yrl + + $i "Generate unrelated .erl files" + $t echo "-module(boy)." > $(APP)/src/boy.erl + $t echo "-module(girl)." > $(APP)/src/girl.erl + + $i "Build the application" + $t $(MAKE) -C $(APP) $v + + $i "Check that all compiled files exist" + $t test -f $(APP)/$(APP).d + $t test -f $(APP)/ebin/$(APP).app + $t test -f $(APP)/ebin/boy.beam + $t test -f $(APP)/ebin/girl.beam + $t test -f $(APP)/ebin/xmerl_xpath_parse.beam + $t test -f $(APP)/ebin/xref_parser.beam + $t test -f $(APP)/src/xmerl_xpath_parse.erl + $t test -f $(APP)/src/xref_parser.erl + + $i "Check that the application was compiled correctly" + $t $(ERL) -pa $(APP)/ebin/ -eval " \ + ok = application:start($(APP)), \ + {ok, Mods = [boy, girl, xmerl_xpath_parse, xref_parser]} \ + = application:get_key($(APP), modules), \ + [{module, M} = code:load_file(M) || M <- Mods], \ + halt()" + + $i "Touch one .yrl file; check that only required files are rebuilt" + $t printf "%s\n" \ + $(APP)/$(APP).d \ + $(APP)/ebin/$(APP).app \ + $(APP)/ebin/xref_parser.beam \ + $(APP)/src/xref_parser.erl | sort > $(APP)/EXPECT + $t touch $(APP)/src/xref_parser.yrl + $t $(MAKE) -C $(APP) $v + $t find $(APP) -type f -newer $(APP)/src/xref_parser.yrl | sort | diff $(APP)/EXPECT - + $t rm $(APP)/EXPECT + + $i "Check that the application was compiled correctly" + $t $(ERL) -pa $(APP)/ebin/ -eval " \ + ok = application:start($(APP)), \ + {ok, Mods = [boy, girl, xmerl_xpath_parse, xref_parser]} \ + = application:get_key($(APP), modules), \ + [{module, M} = code:load_file(M) || M <- Mods], \ + halt()" + + $i "Clean the application" + $t $(MAKE) -C $(APP) clean $v + + $i "Check that source files still exist" + $t test -f $(APP)/Makefile + $t test -f $(APP)/erlang.mk +ifdef LEGACY + $t test -f $(APP)/src/$(APP).app.src +endif + $t test -f $(APP)/src/boy.erl + $t test -f $(APP)/src/girl.erl + $t test -f $(APP)/src/xmerl_xpath_parse.yrl + $t test -f $(APP)/src/xref_parser.yrl + + $i "Check that all build artifacts are removed, including intermediates" + $t test ! -e $(APP)/$(APP).d + $t test ! -e $(APP)/ebin/ + $t test ! -e $(APP)/src/xmerl_xpath_parse.erl + $t test ! -e $(APP)/src/xref_parser.erl + + $i "Build the application again" + $t $(MAKE) -C $(APP) $v + + $i "Check that all compiled files exist" + $t test -f $(APP)/$(APP).d + $t test -f $(APP)/ebin/$(APP).app + $t test -f $(APP)/ebin/boy.beam + $t test -f $(APP)/ebin/girl.beam + $t test -f $(APP)/ebin/xmerl_xpath_parse.beam + $t test -f $(APP)/ebin/xref_parser.beam + $t test -f $(APP)/src/xmerl_xpath_parse.erl + $t test -f $(APP)/src/xref_parser.erl + + $i "Check that the application was compiled correctly" + $t $(ERL) -pa $(APP)/ebin/ -eval " \ + ok = application:start($(APP)), \ + {ok, Mods = [boy, girl, xmerl_xpath_parse, xref_parser]} \ + = application:get_key($(APP), modules), \ + [{module, M} = code:load_file(M) || M <- Mods], \ + halt()" + +core-app-yrl-include: build clean-core-app-yrl-include + + $i "Bootstrap a new OTP library named $(APP)" + $t mkdir $(APP)/ + $t cp ../erlang.mk $(APP)/ + $t $(MAKE) -C $(APP) -f erlang.mk bootstrap-lib $v + + $i "Download a .yrl file with includes from Erlang/OTP" + $t curl -s -o $(APP)/src/core_parse.yrl $(OTP_MASTER)/lib/compiler/src/core_parse.yrl + $t curl -s -o $(APP)/src/core_parse.hrl $(OTP_MASTER)/lib/compiler/src/core_parse.hrl + + $i "Generate unrelated .erl files" + $t echo "-module(boy)." > $(APP)/src/boy.erl + $t echo "-module(girl)." > $(APP)/src/girl.erl + + $i "Build the application" + $t $(MAKE) -C $(APP) $v + + $i "Check that all compiled files exist" + $t test -f $(APP)/$(APP).d + $t test -f $(APP)/ebin/$(APP).app + $t test -f $(APP)/ebin/boy.beam + $t test -f $(APP)/ebin/core_parse.beam + $t test -f $(APP)/ebin/girl.beam + $t test -f $(APP)/src/core_parse.erl + + $i "Check that the application was compiled correctly" + $t $(ERL) -pa $(APP)/ebin/ -eval " \ + ok = application:start($(APP)), \ + {ok, Mods = [boy, core_parse, girl]} \ + = application:get_key($(APP), modules), \ + [{module, M} = code:load_file(M) || M <- Mods], \ + halt()" + + $i "Touch the .yrl file; check that only required files are rebuilt" + $t printf "%s\n" \ + $(APP)/$(APP).d \ + $(APP)/ebin/$(APP).app \ + $(APP)/ebin/core_parse.beam \ + $(APP)/src/core_parse.erl | sort > $(APP)/EXPECT + $t touch $(APP)/src/core_parse.yrl + $t $(MAKE) -C $(APP) $v + $t find $(APP) -type f -newer $(APP)/src/core_parse.yrl | sort | diff $(APP)/EXPECT - + $t rm $(APP)/EXPECT + + $i "Check that the application was compiled correctly" + $t $(ERL) -pa $(APP)/ebin/ -eval " \ + ok = application:start($(APP)), \ + {ok, Mods = [boy, core_parse, girl]} \ + = application:get_key($(APP), modules), \ + [{module, M} = code:load_file(M) || M <- Mods], \ + halt()" + + $i "Touch the .hrl file included; check that only required files are rebuilt" + $t printf "%s\n" \ + $(APP)/$(APP).d \ + $(APP)/ebin/$(APP).app \ + $(APP)/ebin/core_parse.beam \ + $(APP)/src/core_parse.erl | sort > $(APP)/EXPECT + $t touch $(APP)/src/core_parse.hrl + $t $(MAKE) -C $(APP) $v + $t find $(APP) -type f -newer $(APP)/src/core_parse.hrl | sort | diff $(APP)/EXPECT - + $t rm $(APP)/EXPECT + + $i "Check that the application was compiled correctly" + $t $(ERL) -pa $(APP)/ebin/ -eval " \ + ok = application:start($(APP)), \ + {ok, Mods = [boy, core_parse, girl]} \ + = application:get_key($(APP), modules), \ + [{module, M} = code:load_file(M) || M <- Mods], \ + halt()" + + $i "Clean the application" + $t $(MAKE) -C $(APP) clean $v + + $i "Check that source files still exist" + $t test -f $(APP)/Makefile + $t test -f $(APP)/erlang.mk +ifdef LEGACY + $t test -f $(APP)/src/$(APP).app.src +endif + $t test -f $(APP)/src/boy.erl + $t test -f $(APP)/src/core_parse.hrl + $t test -f $(APP)/src/core_parse.yrl + $t test -f $(APP)/src/girl.erl + + $i "Check that all build artifacts are removed, including intermediates" + $t test ! -e $(APP)/$(APP).d + $t test ! -e $(APP)/ebin/ + $t test ! -e $(APP)/src/core_parse.erl + + $i "Build the application again" + $t $(MAKE) -C $(APP) $v + + $i "Check that all compiled files exist" + $t test -f $(APP)/$(APP).d + $t test -f $(APP)/ebin/$(APP).app + $t test -f $(APP)/ebin/boy.beam + $t test -f $(APP)/ebin/core_parse.beam + $t test -f $(APP)/ebin/girl.beam + $t test -f $(APP)/src/core_parse.erl + + $i "Check that the application was compiled correctly" + $t $(ERL) -pa $(APP)/ebin/ -eval " \ + ok = application:start($(APP)), \ + {ok, Mods = [boy, core_parse, girl]} \ + = application:get_key($(APP), modules), \ + [{module, M} = code:load_file(M) || M <- Mods], \ + halt()" diff --git a/test/core_compat.mk b/test/core_compat.mk new file mode 100644 index 0000000..9b6516f --- /dev/null +++ b/test/core_compat.mk @@ -0,0 +1,221 @@ +# Core: Compatibility with other build tools. +# +# Note: autopatch functionality is covered separately. + +CORE_COMPAT_CASES = auto-rebar rebar rebar-deps rebar-deps-pkg rebar-erlc-opts +CORE_COMPAT_TARGETS = $(addprefix core-compat-,$(CORE_COMPAT_CASES)) +CORE_COMPAT_CLEAN_TARGETS = $(addprefix clean-,$(CORE_COMPAT_TARGETS)) + +REBAR_BINARY = https://github.com/rebar/rebar/releases/download/2.6.0/rebar + +.PHONY: core-compat $(CORE_COMPAT_TARGETS) clean-core-compat $(CORE_COMPAT_CLEAN_TARGETS) + +clean-core-compat: $(CORE_COMPAT_CLEAN_TARGETS) + +$(CORE_COMPAT_CLEAN_TARGETS): + $t rm -rf $(APP_TO_CLEAN)/ + +core-compat: $(CORE_COMPAT_TARGETS) + +core-compat-auto-rebar: build clean-core-compat-auto-rebar + + $i "Bootstrap a new OTP library named $(APP)" + $t mkdir $(APP)/ + $t cp ../erlang.mk $(APP)/ + $t $(MAKE) -C $(APP) -f erlang.mk bootstrap-lib $v + + $i "Add rebar.config as a dependency of 'app' target" + $t echo "app:: rebar.config" >> $(APP)/Makefile + + $i "Build the application" + $t $(MAKE) -C $(APP) $v + + $i "Check that rebar.config was created" + $t test -f $(APP)/rebar.config + + $i "Check that rebar.config can be loaded" + $t $(ERL) -eval "{ok, _} = file:consult(\"$(APP)/rebar.config\"), halt()" + + $i "Create a temporary file" + $t touch $(APP)/older_file + + $i "Wait a second" + $t sleep 1 + + $i "Build the application again" + $t $(MAKE) -C $(APP) $v + + $i "Check that rebar.config is newer than the temporary file" + $t test $(APP)/rebar.config -nt $(APP)/older_file + + $i "Check that rebar.config can be loaded" + $t $(ERL) -eval "{ok, _} = file:consult(\"$(APP)/rebar.config\"), halt()" + + $i "Distclean the application" + $t $(MAKE) -C $(APP) distclean $v + + $i "Download rebar" + $t curl -s -L -o $(APP)/rebar $(REBAR_BINARY) + $t chmod +x $(APP)/rebar + + $i "Use rebar to build the application" + $t cd $(APP) && ./rebar compile $v + +core-compat-rebar: build clean-core-compat-rebar + + $i "Bootstrap a new OTP library named $(APP)" + $t mkdir $(APP)/ + $t cp ../erlang.mk $(APP)/ + $t $(MAKE) -C $(APP) -f erlang.mk bootstrap-lib $v + + $i "Run 'make rebar.config'" + $t $(MAKE) -C $(APP) rebar.config $v + + $i "Check that rebar.config was created" + $t test -f $(APP)/rebar.config + + $i "Check that rebar.config can be loaded" + $t $(ERL) -eval "{ok, _} = file:consult(\"$(APP)/rebar.config\"), halt()" + + $i "Create a temporary file" + $t touch $(APP)/older_file + + $i "Wait a second" + $t sleep 1 + + $i "Run 'make rebar.config' again" + $t $(MAKE) -C $(APP) rebar.config $v + + $i "Check that rebar.config is newer than the temporary file" + $t test $(APP)/rebar.config -nt $(APP)/older_file + + $i "Check that rebar.config can be loaded" + $t $(ERL) -eval "{ok, _} = file:consult(\"$(APP)/rebar.config\"), halt()" + + $i "Distclean the application" + $t $(MAKE) -C $(APP) distclean $v + + $i "Download rebar" + $t curl -s -L -o $(APP)/rebar $(REBAR_BINARY) + $t chmod +x $(APP)/rebar + + $i "Use rebar to build the application" + $t cd $(APP) && ./rebar compile $v + +core-compat-rebar-deps: build clean-core-compat-rebar-deps + + $i "Bootstrap a new OTP library named $(APP)" + $t mkdir $(APP)/ + $t cp ../erlang.mk $(APP)/ + $t $(MAKE) -C $(APP) -f erlang.mk bootstrap-lib $v + + $i "Add Cowboy as a dependency" + $t sed -i.bak '2i\ +DEPS = cowboy\ +dep_cowboy = git https://github.com/ninenines/cowboy 1.0.0\ +' $(APP)/Makefile + + $i "Run 'make rebar.config'" + $t $(MAKE) -C $(APP) rebar.config $v + + $i "Check that rebar.config was created" + $t test -f $(APP)/rebar.config + + $i "Check that Cowboy is listed in rebar.config" + $t $(ERL) -eval " \ + {ok, C} = file:consult(\"$(APP)/rebar.config\"), \ + {_, [{cowboy, _, {git, _, \"1.0.0\"}}]} = lists:keyfind(deps, 1, C), \ + halt()" + + $i "Distclean the application" + $t $(MAKE) -C $(APP) distclean $v + + $i "Download rebar" + $t curl -s -L -o $(APP)/rebar $(REBAR_BINARY) + $t chmod +x $(APP)/rebar + + $i "Use rebar to build the application" + $t cd $(APP) && ./rebar get-deps compile $v + +core-compat-rebar-deps-pkg: build clean-core-compat-rebar-deps-pkg + + $i "Bootstrap a new OTP library named $(APP)" + $t mkdir $(APP)/ + $t cp ../erlang.mk $(APP)/ + $t $(MAKE) -C $(APP) -f erlang.mk bootstrap-lib $v + + $i "Add Cowboy package as a dependency" + $t sed -i.bak '2i\ +DEPS = cowboy\ +' $(APP)/Makefile + + $i "Run 'make rebar.config'" + $t $(MAKE) -C $(APP) rebar.config $v + + $i "Check that rebar.config was created" + $t test -f $(APP)/rebar.config + + $i "Check that Cowboy is listed in rebar.config" + $t $(ERL) -eval " \ + {ok, C} = file:consult(\"$(APP)/rebar.config\"), \ + {_, [{cowboy, _, {git, \"https://github.com/\" ++ _, _}}]} = lists:keyfind(deps, 1, C), \ + halt()" + + $i "Distclean the application" + $t $(MAKE) -C $(APP) distclean $v + + $i "Download rebar" + $t curl -s -L -o $(APP)/rebar $(REBAR_BINARY) + $t chmod +x $(APP)/rebar + + $i "Use rebar to build the application" + $t cd $(APP) && ./rebar get-deps compile $v + +core-compat-rebar-erlc-opts: build clean-core-compat-rebar-erlc-opts + + $i "Bootstrap a new OTP library named $(APP)" + $t mkdir $(APP)/ + $t cp ../erlang.mk $(APP)/ + $t $(MAKE) -C $(APP) -f erlang.mk bootstrap-lib $v + + $i "Add extra options to ERLC_OPTS" + $t echo "ERLC_OPTS += +warn_export_all +warn_missing_spec +warn_untyped_record" >> $(APP)/Makefile + + $i "Run 'make rebar.config'" + $t $(MAKE) -C $(APP) rebar.config $v + + $i "Check that rebar.config was created" + $t test -f $(APP)/rebar.config + + $i "Check that -Werror is not listed in rebar.config" + $t $(ERL) -eval " \ + {ok, C} = file:consult(\"$(APP)/rebar.config\"), \ + {_, Opts} = lists:keyfind(erl_opts, 1, C), \ + false = lists:member(warning_as_errors, Opts), \ + halt()" + + $i "Check that debug_info is listed in rebar.config" + $t $(ERL) -eval " \ + {ok, C} = file:consult(\"$(APP)/rebar.config\"), \ + {_, Opts} = lists:keyfind(erl_opts, 1, C), \ + true = lists:member(debug_info, Opts), \ + halt()" + + $i "Check that extra options are listed in rebar.config" + $t $(ERL) -eval " \ + {ok, C} = file:consult(\"$(APP)/rebar.config\"), \ + {_, Opts} = lists:keyfind(erl_opts, 1, C), \ + true = lists:member(warn_export_all, Opts), \ + true = lists:member(warn_missing_spec, Opts), \ + true = lists:member(warn_untyped_record, Opts), \ + halt()" + + $i "Distclean the application" + $t $(MAKE) -C $(APP) distclean $v + + $i "Download rebar" + $t curl -s -L -o $(APP)/rebar $(REBAR_BINARY) + $t chmod +x $(APP)/rebar + + $i "Use rebar to build the application" + $t cd $(APP) && ./rebar compile $v diff --git a/test/core_plugins.mk b/test/core_plugins.mk new file mode 100644 index 0000000..5bbee3f --- /dev/null +++ b/test/core_plugins.mk @@ -0,0 +1,86 @@ +# Core: External plugins. + +CORE_PLUGINS_CASES = all one +CORE_PLUGINS_TARGETS = $(addprefix core-plugins-,$(CORE_PLUGINS_CASES)) +CORE_PLUGINS_CLEAN_TARGETS = $(addprefix clean-,$(CORE_PLUGINS_TARGETS)) + +.PHONY: core-plugins $(CORE_PLUGINS_TARGETS) clean-core-plugins $(CORE_PLUGINS_CLEAN_TARGETS) + +clean-core-plugins: $(CORE_PLUGINS_CLEAN_TARGETS) + +$(CORE_PLUGINS_CLEAN_TARGETS): + $t rm -rf $(APP_TO_CLEAN)/ + +core-plugins: $(CORE_PLUGINS_TARGETS) + +core-plugins-all: build clean-core-plugins-all + + $i "Bootstrap a new OTP library named $(APP)" + $t mkdir $(APP)/ + $t cp ../erlang.mk $(APP)/ + $t $(MAKE) -C $(APP) -f erlang.mk bootstrap-lib $v + + $i "Create a local git repository with two plugins" + $t mkdir -p $(APP)/plugin_dep/mk + $t echo "plugin1: ; @echo \$$@" > $(APP)/plugin_dep/mk/plugin1.mk + $t echo "plugin2: ; @echo \$$@" > $(APP)/plugin_dep/mk/plugin2.mk + $t echo "THIS := \$$(dir \$$(realpath \$$(lastword \$$(MAKEFILE_LIST))))" > $(APP)/plugin_dep/plugins.mk + $t printf "%s\n" "include \$$(THIS)/mk/plugin1.mk" >> $(APP)/plugin_dep/plugins.mk + $t printf "%s\n" "include \$$(THIS)/mk/plugin2.mk" >> $(APP)/plugin_dep/plugins.mk +# We check that overriding THIS doesn't cause an error. + $t echo "THIS :=" >> $(APP)/plugin_dep/plugins.mk + $t cd $(APP)/plugin_dep && \ + git init -q && \ + git config user.email "[email protected]" && \ + git config user.name "test suite" && \ + git add . && \ + git commit -q -m "Tests" + + $i "Add dependency and plugins to the Makefile" + $t sed -i.bak '2i\ +DEPS = plugin_dep\ +dep_plugin_dep = git file://$(abspath $(APP)/plugin_dep) master\ +DEP_PLUGINS = plugin_dep\ +' $(APP)/Makefile + + $i "Run 'make plugin1' and check that it prints plugin1" + $t test -n "`$(MAKE) -C $(APP) plugin1 | grep plugin1`" + + $i "Run 'make plugin2' and check that it prints plugin2" + $t test -n "`$(MAKE) -C $(APP) plugin2 | grep plugin2`" + +core-plugins-one: build clean-core-plugins-one + + $i "Bootstrap a new OTP library named $(APP)" + $t mkdir $(APP)/ + $t cp ../erlang.mk $(APP)/ + $t $(MAKE) -C $(APP) -f erlang.mk bootstrap-lib $v + + $i "Create a local git repository with two plugins" + $t mkdir -p $(APP)/plugin_dep/mk + $t echo "plugin1: ; @echo \$$@" > $(APP)/plugin_dep/mk/plugin1.mk + $t echo "plugin2: ; @echo \$$@" > $(APP)/plugin_dep/mk/plugin2.mk + $t echo "THIS := \$$(dir \$$(realpath \$$(lastword \$$(MAKEFILE_LIST))))" > $(APP)/plugin_dep/plugins.mk + $t printf "%s\n" "include \$$(THIS)/mk/plugin1.mk" >> $(APP)/plugin_dep/plugins.mk + $t printf "%s\n" "include \$$(THIS)/mk/plugin2.mk" >> $(APP)/plugin_dep/plugins.mk +# We check that overriding THIS doesn't cause an error. + $t echo "THIS :=" >> $(APP)/plugin_dep/plugins.mk + $t cd $(APP)/plugin_dep && \ + git init -q && \ + git config user.email "[email protected]" && \ + git config user.name "test suite" && \ + git add . && \ + git commit -q -m "Tests" + + $i "Add dependency and plugins to the Makefile" + $t sed -i.bak '2i\ +DEPS = plugin_dep\ +dep_plugin_dep = git file://$(abspath $(APP)/plugin_dep) master\ +DEP_PLUGINS = plugin_dep/mk/plugin1.mk\ +' $(APP)/Makefile + + $i "Run 'make plugin1' and check that it prints plugin1" + $t test -n "`$(MAKE) -C $(APP) plugin1 | grep plugin1`" + + $i "Run 'make plugin2' and confirm the target doesn't exist" + $t if `$(MAKE) -C $(APP) plugin2`; then false; fi diff --git a/test/core_upgrade.mk b/test/core_upgrade.mk new file mode 100644 index 0000000..ec49a9f --- /dev/null +++ b/test/core_upgrade.mk @@ -0,0 +1,124 @@ +# Core: Erlang.mk upgrade. + +CORE_UPGRADE_CASES = custom-build-dir custom-config custom-repo no-config renamed-config +CORE_UPGRADE_TARGETS = $(addprefix core-upgrade-,$(CORE_UPGRADE_CASES)) +CORE_UPGRADE_CLEAN_TARGETS = $(addprefix clean-,$(CORE_UPGRADE_TARGETS)) + +.PHONY: core-upgrade $(CORE_UPGRADE_TARGETS) clean-core-upgrade $(CORE_UPGRADE_CLEAN_TARGETS) + +clean-core-upgrade: $(CORE_UPGRADE_CLEAN_TARGETS) + +$(CORE_UPGRADE_CLEAN_TARGETS): + $t rm -rf $(APP_TO_CLEAN)/ + +core-upgrade: $(CORE_UPGRADE_TARGETS) + +core-upgrade-custom-build-dir: build clean-core-upgrade-custom-build-dir + + $i "Bootstrap a new OTP library named $(APP)" + $t mkdir $(APP)/ + $t cp ../erlang.mk $(APP)/ + $t $(MAKE) -C $(APP) -f erlang.mk bootstrap-lib $v + + $i "Append a rule to the Erlang.mk file for testing purposes" + $t echo "erlang_mk_upgrade_test_rule: ; @echo FAIL" >> $(APP)/erlang.mk + + $i "Check that the test rule works as intended" + $t test "FAIL" = "`$(MAKE) -C $(APP) --no-print-directory erlang_mk_upgrade_test_rule V=0`" + + $i "Create the custom build directory" + $t mkdir $(APP)/custom/ + $t test -d $(APP)/custom/ + + $i "Upgrade Erlang.mk with a custom build directory" + $t ERLANG_MK_BUILD_DIR=custom $(MAKE) -C $(APP) erlang-mk $v + + $i "Check that the rule is gone" + $t if $(MAKE) -C $(APP) erlang_mk_upgrade_test_rule $v; then false; fi + + $i "Check that the custom build directory is gone" + $t test ! -d $(APP)/custom/ + +core-upgrade-custom-config: build clean-core-upgrade-custom-config + + $i "Bootstrap a new OTP library named $(APP)" + $t mkdir $(APP)/ + $t cp ../erlang.mk $(APP)/ + $t $(MAKE) -C $(APP) -f erlang.mk bootstrap-lib $v + + $i "Create a custom build.config file without plugins" + $t echo "core/*" > $(APP)/build.config + + $i "Upgrade Erlang.mk" + $t $(MAKE) -C $(APP) erlang-mk $v + + $i "Check that the bootstrap plugin is gone" + $t if $(MAKE) -C $(APP) list-templates $v; then false; fi + +core-upgrade-custom-repo: build clean-core-upgrade-custom-repo + + $i "Bootstrap a new OTP library named $(APP)" + $t mkdir $(APP)/ + $t cp ../erlang.mk $(APP)/ + $t $(MAKE) -C $(APP) -f erlang.mk bootstrap-lib $v + + $i "Fork erlang.mk locally and modify it" + $t git clone -q https://github.com/ninenines/erlang.mk $(APP)/alt-erlangmk-repo + $t sed -i.bak '1i\ +# Copyright (c) erlang.mk Testsuite!\ +' $(APP)/alt-erlangmk-repo/core/core.mk + $t (cd $(APP)/alt-erlangmk-repo && \ + git checkout -q -b test-copyright && \ + git config user.email "[email protected]" && \ + git config user.name "test suite" && \ + git commit -q -a -m 'Add Testsuite copyright') + + $i "Point application to an alternate erlang.mk repository" + $t sed -i.bak '2i\ +ERLANG_MK_REPO = file://$(abspath $(APP)/alt-erlangmk-repo)\ +ERLANG_MK_COMMIT = test-copyright\ +' $(APP)/Makefile + + $i "Update erlang.mk" + $t $(MAKE) -C $(APP) erlang-mk $v + + $i "Check our modification is there" + $t grep -q "# Copyright (c) erlang.mk Testsuite!" $(APP)/erlang.mk + +core-upgrade-no-config: build clean-core-upgrade-no-config + + $i "Bootstrap a new OTP library named $(APP)" + $t mkdir $(APP)/ + $t cp ../erlang.mk $(APP)/ + $t $(MAKE) -C $(APP) -f erlang.mk bootstrap-lib $v + + $i "Append a rule to the Erlang.mk file for testing purposes" + $t echo "erlang_mk_upgrade_test_rule: ; @echo FAIL" >> $(APP)/erlang.mk + + $i "Check that the test rule works as intended" + $t test "FAIL" = "`$(MAKE) -C $(APP) --no-print-directory erlang_mk_upgrade_test_rule V=0`" + + $i "Upgrade Erlang.mk" + $t $(MAKE) -C $(APP) erlang-mk $v + + $i "Check that the rule is gone" + $t if $(MAKE) -C $(APP) erlang_mk_upgrade_test_rule $v; then false; fi + +core-upgrade-renamed-config: build clean-core-upgrade-renamed-config + + $i "Bootstrap a new OTP library named $(APP)" + $t mkdir $(APP)/ + $t cp ../erlang.mk $(APP)/ + $t $(MAKE) -C $(APP) -f erlang.mk bootstrap-lib $v + + $i "Create a custom build.config file without plugins; name it my.build.config" + $t echo "core/*" > $(APP)/my.build.config + + $i "Set ERLANG_MK_BUILD_CONFIG=my.build.config in the Makefile" + $t echo "ERLANG_MK_BUILD_CONFIG = my.build.config" >> $(APP)/Makefile + + $i "Upgrade Erlang.mk" + $t $(MAKE) -C $(APP) erlang-mk $v + + $i "Check that the bootstrap plugin is gone" + $t if $(MAKE) -C $(APP) list-templates $v; then false; fi diff --git a/test/plugin_bootstrap.mk b/test/plugin_bootstrap.mk new file mode 100644 index 0000000..16bfb58 --- /dev/null +++ b/test/plugin_bootstrap.mk @@ -0,0 +1,155 @@ +# Bootstrap plugin. + +BOOTSTRAP_CASES = app lib rel templates +BOOTSTRAP_TARGETS = $(addprefix bootstrap-,$(BOOTSTRAP_CASES)) +BOOTSTRAP_CLEAN_TARGETS = $(addprefix clean-,$(BOOTSTRAP_TARGETS)) + +.PHONY: bootstrap $(BOOTSTRAP_TARGETS) clean-bootstrap $(BOOTSTRAP_CLEAN_TARGETS) + +clean-bootstrap: $(BOOTSTRAP_CLEAN_TARGETS) + +$(BOOTSTRAP_CLEAN_TARGETS): + $t rm -rf $(APP_TO_CLEAN)/ + +bootstrap: $(BOOTSTRAP_TARGETS) + +bootstrap-app: build clean-bootstrap-app + + $i "Bootstrap a new OTP application named $(APP)" + $t mkdir $(APP)/ + $t cp ../erlang.mk $(APP)/ + $t $(MAKE) -C $(APP) -f erlang.mk bootstrap $v + + $i "Check that all bootstrapped files exist" + $t test -f $(APP)/Makefile +ifdef LEGACY + $t test -f $(APP)/src/$(APP).app.src +endif + $t test -f $(APP)/src/$(APP)_app.erl + $t test -f $(APP)/src/$(APP)_sup.erl + + $i "Build the application" + $t $(MAKE) -C $(APP) $v + + $i "Check that all compiled files exist" + $t test -f $(APP)/ebin/$(APP).app + $t test -f $(APP)/ebin/$(APP)_app.beam + $t test -f $(APP)/ebin/$(APP)_sup.beam + + $i "Check that the application was compiled correctly" + $t $(ERL) -pa $(APP)/ebin/ -eval " \ + ok = application:start($(APP)), \ + {ok, [$(APP)_app, $(APP)_sup]} = application:get_key($(APP), modules), \ + {module, $(APP)_app} = code:load_file($(APP)_app), \ + {module, $(APP)_sup} = code:load_file($(APP)_sup), \ + halt()" + +bootstrap-lib: build clean-bootstrap-lib + + $i "Bootstrap a new OTP library named $(APP)" + $t mkdir $(APP)/ + $t cp ../erlang.mk $(APP)/ + $t $(MAKE) -C $(APP) -f erlang.mk bootstrap-lib $v + + $i "Check that all bootstrapped files exist" + $t test -f $(APP)/Makefile +ifdef LEGACY + $t test -f $(APP)/src/$(APP).app.src +endif + + $i "Build the application" + $t $(MAKE) -C $(APP) $v + + $i "Check that all compiled files exist" + $t test -f $(APP)/ebin/$(APP).app + + $i "Check that the application was compiled correctly" + $t $(ERL) -pa $(APP)/ebin/ -eval " \ + ok = application:start($(APP)), \ + {ok, []} = application:get_key($(APP), modules), \ + halt()" + +bootstrap-rel: build clean-bootstrap-rel + + $i "Bootstrap a new release-enabled OTP application named $(APP)" + $t mkdir $(APP)/ + $t cp ../erlang.mk $(APP)/ + $t $(MAKE) -C $(APP) -f erlang.mk bootstrap bootstrap-rel $v + + $i "Check that all bootstrapped files exist" + $t test -f $(APP)/Makefile + $t test -f $(APP)/relx.config + $t test -f $(APP)/rel/sys.config + $t test -f $(APP)/rel/vm.args +ifdef LEGACY + $t test -f $(APP)/src/$(APP).app.src +endif + $t test -f $(APP)/src/$(APP)_app.erl + $t test -f $(APP)/src/$(APP)_sup.erl + + $i "Build the application and the release" + $t $(MAKE) -C $(APP) $v + + $i "Check that all compiled files exist" + $t test -f $(APP)/ebin/$(APP).app + $t test -f $(APP)/ebin/$(APP)_app.beam + $t test -f $(APP)/ebin/$(APP)_sup.beam + + $i "Check that the release was generated" +ifeq ($(PLATFORM),msys2) + $t test -f $(APP)/_rel/$(APP)_release/bin/$(APP)_release.cmd +else + $t test -f $(APP)/_rel/$(APP)_release/bin/$(APP)_release +endif + + $i "Check that the release can be started and stopped" +ifeq ($(PLATFORM),msys2) + $t $(APP)/_rel/$(APP)_release/bin/$(APP)_release.cmd install $v + $t $(APP)/_rel/$(APP)_release/bin/$(APP)_release.cmd start $v + $t $(APP)/_rel/$(APP)_release/bin/$(APP)_release.cmd stop $v + $t $(APP)/_rel/$(APP)_release/bin/$(APP)_release.cmd uninstall $v +else + $t $(APP)/_rel/$(APP)_release/bin/$(APP)_release start $v + $t $(APP)/_rel/$(APP)_release/bin/$(APP)_release stop $v +endif + + $i "Check that there's no erl_crash.dump file" + $t test ! -f $(APP)/_rel/$(APP)_release/erl_crash.dump + +bootstrap-templates: build clean-bootstrap-templates + + $i "Bootstrap a new OTP library named $(APP)" + $t mkdir $(APP)/ + $t cp ../erlang.mk $(APP)/ + $t $(MAKE) -C $(APP) -f erlang.mk bootstrap-lib $v + + $i "Check that we can get the list of templates" + $t test `$(MAKE) -C $(APP) --no-print-directory list-templates V=0 | wc -l` -eq 1 + + $i "Generate one of each template" + $t $(MAKE) -C $(APP) --no-print-directory new t=gen_fsm n=my_fsm + $t $(MAKE) -C $(APP) --no-print-directory new t=gen_server n=my_server + $t $(MAKE) -C $(APP) --no-print-directory new t=supervisor n=my_sup + $t $(MAKE) -C $(APP) --no-print-directory new t=cowboy_http n=my_http + $t $(MAKE) -C $(APP) --no-print-directory new t=cowboy_loop n=my_loop + $t $(MAKE) -C $(APP) --no-print-directory new t=cowboy_rest n=my_rest + $t $(MAKE) -C $(APP) --no-print-directory new t=cowboy_ws n=my_ws + $t $(MAKE) -C $(APP) --no-print-directory new t=ranch_protocol n=my_protocol + +# Here we disable warnings because templates contain missing behaviors. + $i "Build the application" + $t $(MAKE) -C $(APP) ERLC_OPTS=+debug_info $v + + $i "Check that all compiled files exist" + $t test -f $(APP)/ebin/$(APP).app + $t test -f $(APP)/ebin/my_fsm.beam + $t test -f $(APP)/ebin/my_server.beam + $t test -f $(APP)/ebin/my_sup.beam + + $i "Check that all the modules can be loaded" + $t $(ERL) -pa $(APP)/ebin/ -eval " \ + ok = application:start($(APP)), \ + {ok, Mods = [my_fsm, my_http, my_loop, my_protocol, my_rest, my_server, my_sup, my_ws]} \ + = application:get_key($(APP), modules), \ + [{module, M} = code:load_file(M) || M <- Mods], \ + halt()" |