deps.mk 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163
  1. # Copyright (c) 2013-2015, Loïc Hoguin <essen@ninenines.eu>
  2. # This file is part of erlang.mk and subject to the terms of the ISC License.
  3. .PHONY: distclean-deps distclean-pkg pkg-list pkg-search
  4. # Configuration.
  5. AUTOPATCH ?= edown gen_leader gproc
  6. export AUTOPATCH
  7. DEPS_DIR ?= $(CURDIR)/deps
  8. export DEPS_DIR
  9. REBAR_DEPS_DIR = $(DEPS_DIR)
  10. export REBAR_DEPS_DIR
  11. ALL_DEPS_DIRS = $(addprefix $(DEPS_DIR)/,$(DEPS))
  12. ifeq ($(filter $(DEPS_DIR),$(subst :, ,$(ERL_LIBS))),)
  13. ifeq ($(ERL_LIBS),)
  14. ERL_LIBS = $(DEPS_DIR)
  15. else
  16. ERL_LIBS := $(ERL_LIBS):$(DEPS_DIR)
  17. endif
  18. endif
  19. export ERL_LIBS
  20. PKG_FILE2 ?= $(CURDIR)/.erlang.mk.packages.v2
  21. export PKG_FILE2
  22. PKG_FILE_URL ?= https://raw.githubusercontent.com/ninenines/erlang.mk/master/packages.v2.tsv
  23. # Core targets.
  24. deps:: $(ALL_DEPS_DIRS)
  25. @for dep in $(ALL_DEPS_DIRS) ; do \
  26. if [ -f $$dep/GNUmakefile ] || [ -f $$dep/makefile ] || [ -f $$dep/Makefile ] ; then \
  27. $(MAKE) -C $$dep ; \
  28. else \
  29. echo "include $(CURDIR)/erlang.mk" | ERLC_OPTS=+debug_info $(MAKE) -f - -C $$dep ; \
  30. fi ; \
  31. done
  32. distclean:: distclean-deps distclean-pkg
  33. # Deps related targets.
  34. define dep_autopatch
  35. $(ERL) -eval " \
  36. DepDir = \"$(DEPS_DIR)/$(1)/\", \
  37. fun() -> \
  38. {ok, Conf} = file:consult(DepDir ++ \"rebar.config\"), \
  39. File = case lists:keyfind(deps, 1, Conf) of false -> []; {_, Deps} -> \
  40. [begin {Method, Repo, Commit} = case Repos of \
  41. {git, R} -> {git, R, master}; \
  42. {M, R, {branch, C}} -> {M, R, C}; \
  43. {M, R, {tag, C}} -> {M, R, C}; \
  44. {M, R, C} -> {M, R, C} \
  45. end, \
  46. io_lib:format(\"DEPS += ~s\ndep_~s = ~s ~s ~s~n\", [Name, Name, Method, Repo, Commit]) \
  47. end || {Name, _, Repos} <- Deps] \
  48. end, \
  49. ok = file:write_file(\"$(DEPS_DIR)/$(1)/Makefile\", [\"ERLC_OPTS = +debug_info\n\n\", File, \"\ninclude erlang.mk\"]) \
  50. end(), \
  51. AppSrcOut = \"$(DEPS_DIR)/$(1)/src/$(1).app.src\", \
  52. AppSrcIn = case filelib:is_regular(AppSrcOut) of false -> \"$(DEPS_DIR)/$(1)/ebin/$(1).app\"; true -> AppSrcOut end, \
  53. fun() -> \
  54. {ok, [{application, $(1), L}]} = file:consult(AppSrcIn), \
  55. L2 = case lists:keyfind(modules, 1, L) of {_, _} -> L; false -> [{modules, []}|L] end, \
  56. L3 = case lists:keyfind(vsn, 1, L2) of {vsn, git} -> lists:keyreplace(vsn, 1, L2, {vsn, \"git\"}); _ -> L2 end, \
  57. ok = file:write_file(AppSrcOut, io_lib:format(\"~p.~n\", [{application, $(1), L3}])) \
  58. end(), \
  59. case AppSrcOut of AppSrcIn -> ok; _ -> ok = file:delete(AppSrcIn) end, \
  60. halt()."
  61. endef
  62. ifeq ($(V),0)
  63. define dep_autopatch_verbose
  64. @echo " PATCH " $(1);
  65. endef
  66. endif
  67. define dep_fetch
  68. if [ "$$$$VS" = "git" ]; then \
  69. git clone -n -- $$$$REPO $(DEPS_DIR)/$(1); \
  70. cd $(DEPS_DIR)/$(1) && git checkout -q $$$$COMMIT; \
  71. elif [ "$$$$VS" = "hg" ]; then \
  72. hg clone -U $$$$REPO $(DEPS_DIR)/$(1); \
  73. cd $(DEPS_DIR)/$(1) && hg update -q $$$$COMMIT; \
  74. elif [ "$$$$VS" = "svn" ]; then \
  75. svn checkout $$$$REPO $(DEPS_DIR)/$(1); \
  76. else \
  77. echo "Unknown or invalid dependency: $(1). Please consult the erlang.mk README for instructions." >&2; \
  78. exit 78; \
  79. fi
  80. endef
  81. define dep_target
  82. $(DEPS_DIR)/$(1):
  83. @mkdir -p $(DEPS_DIR)
  84. ifeq (,$(dep_$(1)))
  85. @if [ ! -f $(PKG_FILE2) ]; then $(call core_http_get,$(PKG_FILE2),$(PKG_FILE_URL)); fi
  86. @DEPPKG=$$$$(awk 'BEGIN { FS = "\t" }; $$$$1 == "$(1)" { print $$$$2 " " $$$$3 " " $$$$4 }' $(PKG_FILE2);); \
  87. VS=$$$$(echo $$$$DEPPKG | cut -d " " -f1); \
  88. REPO=$$$$(echo $$$$DEPPKG | cut -d " " -f2); \
  89. COMMIT=$$$$(echo $$$$DEPPKG | cut -d " " -f3); \
  90. $(call dep_fetch,$(1))
  91. else
  92. @VS=$(word 1,$(dep_$(1))); \
  93. REPO=$(word 2,$(dep_$(1))); \
  94. COMMIT=$(word 3,$(dep_$(1))); \
  95. $(call dep_fetch,$(1))
  96. endif
  97. ifneq ($(filter $(1),$(AUTOPATCH)),)
  98. $(call dep_autopatch_verbose,$(1)) if [ -f $(DEPS_DIR)/$(1)/rebar.config ]; then \
  99. $(call dep_autopatch,$(1)); \
  100. cd $(DEPS_DIR)/$(1)/ && ln -s ../../erlang.mk; \
  101. elif [ ! -f $(DEPS_DIR)/$(1)/Makefile ]; then \
  102. echo "ERLC_OPTS = +debug_info\ninclude erlang.mk" > $(DEPS_DIR)/$(1)/Makefile; \
  103. cd $(DEPS_DIR)/$(1)/ && ln -s ../../erlang.mk; \
  104. fi
  105. endif
  106. endef
  107. $(foreach dep,$(DEPS),$(eval $(call dep_target,$(dep))))
  108. distclean-deps:
  109. $(gen_verbose) rm -rf $(DEPS_DIR)
  110. # Packages related targets.
  111. $(PKG_FILE2):
  112. @$(call core_http_get,$(PKG_FILE2),$(PKG_FILE_URL))
  113. pkg-list: $(PKG_FILE2)
  114. @cat $(PKG_FILE2) | awk 'BEGIN { FS = "\t" }; { print \
  115. "Name:\t\t" $$1 "\n" \
  116. "Repository:\t" $$3 "\n" \
  117. "Website:\t" $$5 "\n" \
  118. "Description:\t" $$6 "\n" }'
  119. ifdef q
  120. pkg-search: $(PKG_FILE2)
  121. @cat $(PKG_FILE2) | grep -i ${q} | awk 'BEGIN { FS = "\t" }; { print \
  122. "Name:\t\t" $$1 "\n" \
  123. "Repository:\t" $$3 "\n" \
  124. "Website:\t" $$5 "\n" \
  125. "Description:\t" $$6 "\n" }'
  126. else
  127. pkg-search:
  128. $(error Usage: make pkg-search q=STRING)
  129. endif
  130. ifeq ($(PKG_FILE2),$(CURDIR)/.erlang.mk.packages.v2)
  131. distclean-pkg:
  132. $(gen_verbose) rm -f $(PKG_FILE2)
  133. endif
  134. help::
  135. @printf "%s\n" "" \
  136. "Package-related targets:" \
  137. " pkg-list List all known packages" \
  138. " pkg-search q=STRING Search for STRING in the package index"