| # Copyright (c) 2013-2016, Loïc Hoguin <essen@ninenines.eu> |
| # |
| # Permission to use, copy, modify, and/or distribute this software for any |
| # purpose with or without fee is hereby granted, provided that the above |
| # copyright notice and this permission notice appear in all copies. |
| # |
| # THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES |
| # WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF |
| # MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR |
| # ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES |
| # WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN |
| # ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF |
| # OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. |
| |
| .PHONY: all app apps deps search rel relup docs install-docs check tests clean distclean help erlang-mk |
| |
| ERLANG_MK_FILENAME := $(realpath $(lastword $(MAKEFILE_LIST))) |
| export ERLANG_MK_FILENAME |
| |
| ERLANG_MK_VERSION = d3485e7 |
| ERLANG_MK_WITHOUT = |
| |
| # Make 3.81 and 3.82 are deprecated. |
| |
| ifeq ($(MAKELEVEL)$(MAKE_VERSION),03.81) |
| $(warning Please upgrade to GNU Make 4 or later: https://erlang.mk/guide/installation.html) |
| endif |
| |
| ifeq ($(MAKELEVEL)$(MAKE_VERSION),03.82) |
| $(warning Please upgrade to GNU Make 4 or later: https://erlang.mk/guide/installation.html) |
| endif |
| |
| # Core configuration. |
| |
| PROJECT ?= $(notdir $(CURDIR)) |
| PROJECT := $(strip $(PROJECT)) |
| |
| PROJECT_VERSION ?= rolling |
| PROJECT_MOD ?= $(PROJECT)_app |
| PROJECT_ENV ?= [] |
| |
| # Verbosity. |
| |
| V ?= 0 |
| |
| verbose_0 = @ |
| verbose_2 = set -x; |
| verbose = $(verbose_$(V)) |
| |
| ifeq ($(V),3) |
| SHELL := $(SHELL) -x |
| endif |
| |
| gen_verbose_0 = @echo " GEN " $@; |
| gen_verbose_2 = set -x; |
| gen_verbose = $(gen_verbose_$(V)) |
| |
| gen_verbose_esc_0 = @echo " GEN " $$@; |
| gen_verbose_esc_2 = set -x; |
| gen_verbose_esc = $(gen_verbose_esc_$(V)) |
| |
| # Temporary files directory. |
| |
| ERLANG_MK_TMP ?= $(CURDIR)/.erlang.mk |
| export ERLANG_MK_TMP |
| |
| # "erl" command. |
| |
| ERL = erl +A1 -noinput -boot no_dot_erlang |
| |
| # Platform detection. |
| |
| ifeq ($(PLATFORM),) |
| UNAME_S := $(shell uname -s) |
| |
| ifeq ($(UNAME_S),Linux) |
| PLATFORM = linux |
| else ifeq ($(UNAME_S),Darwin) |
| PLATFORM = darwin |
| else ifeq ($(UNAME_S),SunOS) |
| PLATFORM = solaris |
| else ifeq ($(UNAME_S),GNU) |
| PLATFORM = gnu |
| else ifeq ($(UNAME_S),FreeBSD) |
| PLATFORM = freebsd |
| else ifeq ($(UNAME_S),NetBSD) |
| PLATFORM = netbsd |
| else ifeq ($(UNAME_S),OpenBSD) |
| PLATFORM = openbsd |
| else ifeq ($(UNAME_S),DragonFly) |
| PLATFORM = dragonfly |
| else ifeq ($(shell uname -o),Msys) |
| PLATFORM = msys2 |
| else |
| $(error Unable to detect platform. Please open a ticket with the output of uname -a.) |
| endif |
| |
| export PLATFORM |
| endif |
| |
| # Core targets. |
| |
| all:: deps app rel |
| |
| # Noop to avoid a Make warning when there's nothing to do. |
| rel:: |
| $(verbose) : |
| |
| relup:: deps app |
| |
| check:: tests |
| |
| clean:: clean-crashdump |
| |
| clean-crashdump: |
| ifneq ($(wildcard erl_crash.dump),) |
| $(gen_verbose) rm -f erl_crash.dump |
| endif |
| |
| distclean:: clean distclean-tmp |
| |
| $(ERLANG_MK_TMP): |
| $(verbose) mkdir -p $(ERLANG_MK_TMP) |
| |
| distclean-tmp: |
| $(gen_verbose) rm -rf $(ERLANG_MK_TMP) |
| |
| help:: |
| $(verbose) printf "%s\n" \ |
| "erlang.mk (version $(ERLANG_MK_VERSION)) is distributed under the terms of the ISC License." \ |
| "Copyright (c) 2013-2016 Loïc Hoguin <essen@ninenines.eu>" \ |
| "" \ |
| "Usage: [V=1] $(MAKE) [target]..." \ |
| "" \ |
| "Core targets:" \ |
| " all Run deps, app and rel targets in that order" \ |
| " app Compile the project" \ |
| " deps Fetch dependencies (if needed) and compile them" \ |
| " fetch-deps Fetch dependencies recursively (if needed) without compiling them" \ |
| " list-deps List dependencies recursively on stdout" \ |
| " search q=... Search for a package in the built-in index" \ |
| " rel Build a release for this project, if applicable" \ |
| " docs Build the documentation for this project" \ |
| " install-docs Install the man pages for this project" \ |
| " check Compile and run all tests and analysis for this project" \ |
| " tests Run the tests for this project" \ |
| " clean Delete temporary and output files from most targets" \ |
| " distclean Delete all temporary and output files" \ |
| " help Display this help and exit" \ |
| " erlang-mk Update erlang.mk to the latest version" |
| |
| # Core functions. |
| |
| empty := |
| space := $(empty) $(empty) |
| tab := $(empty) $(empty) |
| comma := , |
| |
| define newline |
| |
| |
| endef |
| |
| define comma_list |
| $(subst $(space),$(comma),$(strip $(1))) |
| endef |
| |
| define escape_dquotes |
| $(subst ",\",$1) |
| endef |
| |
| # Adding erlang.mk to make Erlang scripts who call init:get_plain_arguments() happy. |
| define erlang |
| $(ERL) $2 -pz $(ERLANG_MK_TMP)/rebar3/_build/prod/lib/*/ebin/ -eval "$(subst $(newline),,$(call escape_dquotes,$1))" -- erlang.mk |
| endef |
| |
| ifeq ($(PLATFORM),msys2) |
| core_native_path = $(shell cygpath -m $1) |
| else |
| core_native_path = $1 |
| endif |
| |
| core_http_get = curl -Lf$(if $(filter-out 0,$(V)),,s)o $(call core_native_path,$1) $2 |
| |
| core_eq = $(and $(findstring $(1),$(2)),$(findstring $(2),$(1))) |
| |
| # We skip files that contain spaces because they end up causing issues. |
| # Files that begin with a dot are already ignored by the wildcard function. |
| core_find = $(foreach f,$(wildcard $(1:%/=%)/*),$(if $(wildcard $f/.),$(call core_find,$f,$2),$(if $(filter $(subst *,%,$2),$f),$(if $(wildcard $f),$f)))) |
| |
| core_lc = $(subst A,a,$(subst B,b,$(subst C,c,$(subst D,d,$(subst E,e,$(subst F,f,$(subst G,g,$(subst H,h,$(subst I,i,$(subst J,j,$(subst K,k,$(subst L,l,$(subst M,m,$(subst N,n,$(subst O,o,$(subst P,p,$(subst Q,q,$(subst R,r,$(subst S,s,$(subst T,t,$(subst U,u,$(subst V,v,$(subst W,w,$(subst X,x,$(subst Y,y,$(subst Z,z,$(1))))))))))))))))))))))))))) |
| |
| core_ls = $(filter-out $(1),$(shell echo $(1))) |
| |
| # @todo Use a solution that does not require using perl. |
| core_relpath = $(shell perl -e 'use File::Spec; print File::Spec->abs2rel(@ARGV) . "\n"' $1 $2) |
| |
| define core_render |
| printf -- '$(subst $(newline),\n,$(subst %,%%,$(subst ','\'',$(subst $(tab),$(WS),$(call $(1))))))\n' > $(2) |
| endef |
| |
| # Automated update. |
| |
| ERLANG_MK_REPO ?= https://github.com/ninenines/erlang.mk |
| ERLANG_MK_COMMIT ?= |
| ERLANG_MK_BUILD_CONFIG ?= build.config |
| ERLANG_MK_BUILD_DIR ?= .erlang.mk.build |
| |
| erlang-mk: WITHOUT ?= $(ERLANG_MK_WITHOUT) |
| erlang-mk: |
| ifdef ERLANG_MK_COMMIT |
| $(verbose) git clone $(ERLANG_MK_REPO) $(ERLANG_MK_BUILD_DIR) |
| $(verbose) cd $(ERLANG_MK_BUILD_DIR) && git checkout $(ERLANG_MK_COMMIT) |
| else |
| $(verbose) git clone --depth 1 $(ERLANG_MK_REPO) $(ERLANG_MK_BUILD_DIR) |
| endif |
| $(verbose) if [ -f $(ERLANG_MK_BUILD_CONFIG) ]; then cp $(ERLANG_MK_BUILD_CONFIG) $(ERLANG_MK_BUILD_DIR)/build.config; fi |
| $(gen_verbose) $(MAKE) --no-print-directory -C $(ERLANG_MK_BUILD_DIR) WITHOUT='$(strip $(WITHOUT))' UPGRADE=1 |
| $(verbose) cp $(ERLANG_MK_BUILD_DIR)/erlang.mk ./erlang.mk |
| $(verbose) rm -rf $(ERLANG_MK_BUILD_DIR) |
| $(verbose) rm -rf $(ERLANG_MK_TMP) |
| |
| # The erlang.mk package index is bundled in the default erlang.mk build. |
| # Search for the string "copyright" to skip to the rest of the code. |
| |
| # Copyright (c) 2015-2017, Loïc Hoguin <essen@ninenines.eu> |
| # This file is part of erlang.mk and subject to the terms of the ISC License. |
| |
| .PHONY: distclean-kerl |
| |
| KERL_INSTALL_DIR ?= $(HOME)/erlang |
| |
| ifeq ($(strip $(KERL)),) |
| KERL := $(ERLANG_MK_TMP)/kerl/kerl |
| endif |
| |
| KERL_DIR = $(ERLANG_MK_TMP)/kerl |
| |
| export KERL |
| |
| KERL_GIT ?= https://github.com/kerl/kerl |
| KERL_COMMIT ?= master |
| |
| KERL_MAKEFLAGS ?= |
| |
| OTP_GIT ?= https://github.com/erlang/otp |
| |
| define kerl_otp_target |
| $(KERL_INSTALL_DIR)/$(1): $(KERL) |
| $(verbose) if [ ! -d $$@ ]; then \ |
| MAKEFLAGS="$(KERL_MAKEFLAGS)" $(KERL) build git $(OTP_GIT) $(1) $(1); \ |
| $(KERL) install $(1) $(KERL_INSTALL_DIR)/$(1); \ |
| fi |
| endef |
| |
| $(KERL): $(KERL_DIR) |
| |
| $(KERL_DIR): | $(ERLANG_MK_TMP) |
| $(gen_verbose) git clone --depth 1 $(KERL_GIT) $(ERLANG_MK_TMP)/kerl |
| $(verbose) cd $(ERLANG_MK_TMP)/kerl && git checkout $(KERL_COMMIT) |
| $(verbose) chmod +x $(KERL) |
| |
| distclean:: distclean-kerl |
| |
| distclean-kerl: |
| $(gen_verbose) rm -rf $(KERL_DIR) |
| |
| # Allow users to select which version of Erlang/OTP to use for a project. |
| |
| ifneq ($(strip $(LATEST_ERLANG_OTP)),) |
| # In some environments it is necessary to filter out master. |
| ERLANG_OTP := $(notdir $(lastword $(sort\ |
| $(filter-out $(KERL_INSTALL_DIR)/master $(KERL_INSTALL_DIR)/OTP_R%,\ |
| $(filter-out %-rc1 %-rc2 %-rc3,$(wildcard $(KERL_INSTALL_DIR)/*[^-native])))))) |
| endif |
| |
| ERLANG_OTP ?= |
| |
| # Use kerl to enforce a specific Erlang/OTP version for a project. |
| ifneq ($(strip $(ERLANG_OTP)),) |
| |
| export PATH := $(KERL_INSTALL_DIR)/$(ERLANG_OTP)/bin:$(PATH) |
| SHELL := env PATH=$(PATH) $(SHELL) |
| $(eval $(call kerl_otp_target,$(ERLANG_OTP))) |
| |
| # Build Erlang/OTP only if it doesn't already exist. |
| ifeq ($(wildcard $(KERL_INSTALL_DIR)/$(ERLANG_OTP))$(BUILD_ERLANG_OTP),) |
| $(info Building Erlang/OTP $(ERLANG_OTP)... Please wait...) |
| $(shell $(MAKE) $(KERL_INSTALL_DIR)/$(ERLANG_OTP) ERLANG_OTP=$(ERLANG_OTP) BUILD_ERLANG_OTP=1 >&2) |
| endif |
| |
| endif |
| |
| PACKAGES += aberth |
| pkg_aberth_name = aberth |
| pkg_aberth_description = Generic BERT-RPC server in Erlang |
| pkg_aberth_homepage = https://github.com/a13x/aberth |
| pkg_aberth_fetch = git |
| pkg_aberth_repo = https://github.com/a13x/aberth |
| pkg_aberth_commit = master |
| |
| PACKAGES += active |
| pkg_active_name = active |
| pkg_active_description = Active development for Erlang: rebuild and reload source/binary files while the VM is running |
| pkg_active_homepage = https://github.com/proger/active |
| pkg_active_fetch = git |
| pkg_active_repo = https://github.com/proger/active |
| pkg_active_commit = master |
| |
| PACKAGES += aleppo |
| pkg_aleppo_name = aleppo |
| pkg_aleppo_description = Alternative Erlang Pre-Processor |
| pkg_aleppo_homepage = https://github.com/ErlyORM/aleppo |
| pkg_aleppo_fetch = git |
| pkg_aleppo_repo = https://github.com/ErlyORM/aleppo |
| pkg_aleppo_commit = master |
| |
| PACKAGES += alog |
| pkg_alog_name = alog |
| pkg_alog_description = Simply the best logging framework for Erlang |
| pkg_alog_homepage = https://github.com/siberian-fast-food/alogger |
| pkg_alog_fetch = git |
| pkg_alog_repo = https://github.com/siberian-fast-food/alogger |
| pkg_alog_commit = master |
| |
| PACKAGES += annotations |
| pkg_annotations_name = annotations |
| pkg_annotations_description = Simple code instrumentation utilities |
| pkg_annotations_homepage = https://github.com/hyperthunk/annotations |
| pkg_annotations_fetch = git |
| pkg_annotations_repo = https://github.com/hyperthunk/annotations |
| pkg_annotations_commit = master |
| |
| PACKAGES += apns |
| pkg_apns_name = apns |
| pkg_apns_description = Apple Push Notification Server for Erlang |
| pkg_apns_homepage = http://inaka.github.com/apns4erl |
| pkg_apns_fetch = git |
| pkg_apns_repo = https://github.com/inaka/apns4erl |
| pkg_apns_commit = master |
| |
| PACKAGES += asciideck |
| pkg_asciideck_name = asciideck |
| pkg_asciideck_description = Asciidoc for Erlang. |
| pkg_asciideck_homepage = https://ninenines.eu |
| pkg_asciideck_fetch = git |
| pkg_asciideck_repo = https://github.com/ninenines/asciideck |
| pkg_asciideck_commit = master |
| |
| PACKAGES += backoff |
| pkg_backoff_name = backoff |
| pkg_backoff_description = Simple exponential backoffs in Erlang |
| pkg_backoff_homepage = https://github.com/ferd/backoff |
| pkg_backoff_fetch = git |
| pkg_backoff_repo = https://github.com/ferd/backoff |
| pkg_backoff_commit = master |
| |
| PACKAGES += barrel_tcp |
| pkg_barrel_tcp_name = barrel_tcp |
| pkg_barrel_tcp_description = barrel is a generic TCP acceptor pool with low latency in Erlang. |
| pkg_barrel_tcp_homepage = https://github.com/benoitc-attic/barrel_tcp |
| pkg_barrel_tcp_fetch = git |
| pkg_barrel_tcp_repo = https://github.com/benoitc-attic/barrel_tcp |
| pkg_barrel_tcp_commit = master |
| |
| PACKAGES += basho_bench |
| pkg_basho_bench_name = basho_bench |
| pkg_basho_bench_description = A load-generation and testing tool for basically whatever you can write a returning Erlang function for. |
| pkg_basho_bench_homepage = https://github.com/basho/basho_bench |
| pkg_basho_bench_fetch = git |
| pkg_basho_bench_repo = https://github.com/basho/basho_bench |
| pkg_basho_bench_commit = master |
| |
| PACKAGES += bcrypt |
| pkg_bcrypt_name = bcrypt |
| pkg_bcrypt_description = Bcrypt Erlang / C library |
| pkg_bcrypt_homepage = https://github.com/erlangpack/bcrypt |
| pkg_bcrypt_fetch = git |
| pkg_bcrypt_repo = https://github.com/erlangpack/bcrypt.git |
| pkg_bcrypt_commit = master |
| |
| PACKAGES += beam |
| pkg_beam_name = beam |
| pkg_beam_description = BEAM emulator written in Erlang |
| pkg_beam_homepage = https://github.com/tonyrog/beam |
| pkg_beam_fetch = git |
| pkg_beam_repo = https://github.com/tonyrog/beam |
| pkg_beam_commit = master |
| |
| PACKAGES += bear |
| pkg_bear_name = bear |
| pkg_bear_description = a set of statistics functions for erlang |
| pkg_bear_homepage = https://github.com/boundary/bear |
| pkg_bear_fetch = git |
| pkg_bear_repo = https://github.com/boundary/bear |
| pkg_bear_commit = master |
| |
| PACKAGES += bertconf |
| pkg_bertconf_name = bertconf |
| pkg_bertconf_description = Make ETS tables out of statc BERT files that are auto-reloaded |
| pkg_bertconf_homepage = https://github.com/ferd/bertconf |
| pkg_bertconf_fetch = git |
| pkg_bertconf_repo = https://github.com/ferd/bertconf |
| pkg_bertconf_commit = master |
| |
| PACKAGES += bifrost |
| pkg_bifrost_name = bifrost |
| pkg_bifrost_description = Erlang FTP Server Framework |
| pkg_bifrost_homepage = https://github.com/thorstadt/bifrost |
| pkg_bifrost_fetch = git |
| pkg_bifrost_repo = https://github.com/thorstadt/bifrost |
| pkg_bifrost_commit = master |
| |
| PACKAGES += binpp |
| pkg_binpp_name = binpp |
| pkg_binpp_description = Erlang Binary Pretty Printer |
| pkg_binpp_homepage = https://github.com/jtendo/binpp |
| pkg_binpp_fetch = git |
| pkg_binpp_repo = https://github.com/jtendo/binpp |
| pkg_binpp_commit = master |
| |
| PACKAGES += bisect |
| pkg_bisect_name = bisect |
| pkg_bisect_description = Ordered fixed-size binary dictionary in Erlang |
| pkg_bisect_homepage = https://github.com/knutin/bisect |
| pkg_bisect_fetch = git |
| pkg_bisect_repo = https://github.com/knutin/bisect |
| pkg_bisect_commit = master |
| |
| PACKAGES += bitcask |
| pkg_bitcask_name = bitcask |
| pkg_bitcask_description = because you need another a key/value storage engine |
| pkg_bitcask_homepage = https://github.com/basho/bitcask |
| pkg_bitcask_fetch = git |
| pkg_bitcask_repo = https://github.com/basho/bitcask |
| pkg_bitcask_commit = develop |
| |
| PACKAGES += bootstrap |
| pkg_bootstrap_name = bootstrap |
| pkg_bootstrap_description = A simple, yet powerful Erlang cluster bootstrapping application. |
| pkg_bootstrap_homepage = https://github.com/schlagert/bootstrap |
| pkg_bootstrap_fetch = git |
| pkg_bootstrap_repo = https://github.com/schlagert/bootstrap |
| pkg_bootstrap_commit = master |
| |
| PACKAGES += boss |
| pkg_boss_name = boss |
| pkg_boss_description = Erlang web MVC, now featuring Comet |
| pkg_boss_homepage = https://github.com/ChicagoBoss/ChicagoBoss |
| pkg_boss_fetch = git |
| pkg_boss_repo = https://github.com/ChicagoBoss/ChicagoBoss |
| pkg_boss_commit = master |
| |
| PACKAGES += boss_db |
| pkg_boss_db_name = boss_db |
| pkg_boss_db_description = BossDB: a sharded, caching, pooling, evented ORM for Erlang |
| pkg_boss_db_homepage = https://github.com/ErlyORM/boss_db |
| pkg_boss_db_fetch = git |
| pkg_boss_db_repo = https://github.com/ErlyORM/boss_db |
| pkg_boss_db_commit = master |
| |
| PACKAGES += brod |
| pkg_brod_name = brod |
| pkg_brod_description = Kafka client in Erlang |
| pkg_brod_homepage = https://github.com/klarna/brod |
| pkg_brod_fetch = git |
| pkg_brod_repo = https://github.com/klarna/brod.git |
| pkg_brod_commit = master |
| |
| PACKAGES += bson |
| pkg_bson_name = bson |
| pkg_bson_description = BSON documents in Erlang, see bsonspec.org |
| pkg_bson_homepage = https://github.com/comtihon/bson-erlang |
| pkg_bson_fetch = git |
| pkg_bson_repo = https://github.com/comtihon/bson-erlang |
| pkg_bson_commit = master |
| |
| PACKAGES += bullet |
| pkg_bullet_name = bullet |
| pkg_bullet_description = Simple, reliable, efficient streaming for Cowboy. |
| pkg_bullet_homepage = http://ninenines.eu |
| pkg_bullet_fetch = git |
| pkg_bullet_repo = https://github.com/ninenines/bullet |
| pkg_bullet_commit = master |
| |
| PACKAGES += cache |
| pkg_cache_name = cache |
| pkg_cache_description = Erlang in-memory cache |
| pkg_cache_homepage = https://github.com/fogfish/cache |
| pkg_cache_fetch = git |
| pkg_cache_repo = https://github.com/fogfish/cache |
| pkg_cache_commit = master |
| |
| PACKAGES += cake |
| pkg_cake_name = cake |
| pkg_cake_description = Really simple terminal colorization |
| pkg_cake_homepage = https://github.com/darach/cake-erl |
| pkg_cake_fetch = git |
| pkg_cake_repo = https://github.com/darach/cake-erl |
| pkg_cake_commit = master |
| |
| PACKAGES += cberl |
| pkg_cberl_name = cberl |
| pkg_cberl_description = NIF based Erlang bindings for Couchbase |
| pkg_cberl_homepage = https://github.com/chitika/cberl |
| pkg_cberl_fetch = git |
| pkg_cberl_repo = https://github.com/chitika/cberl |
| pkg_cberl_commit = master |
| |
| PACKAGES += cecho |
| pkg_cecho_name = cecho |
| pkg_cecho_description = An ncurses library for Erlang |
| pkg_cecho_homepage = https://github.com/mazenharake/cecho |
| pkg_cecho_fetch = git |
| pkg_cecho_repo = https://github.com/mazenharake/cecho |
| pkg_cecho_commit = master |
| |
| PACKAGES += cferl |
| pkg_cferl_name = cferl |
| pkg_cferl_description = Rackspace / Open Stack Cloud Files Erlang Client |
| pkg_cferl_homepage = https://github.com/ddossot/cferl |
| pkg_cferl_fetch = git |
| pkg_cferl_repo = https://github.com/ddossot/cferl |
| pkg_cferl_commit = master |
| |
| PACKAGES += chaos_monkey |
| pkg_chaos_monkey_name = chaos_monkey |
| pkg_chaos_monkey_description = This is The CHAOS MONKEY. It will kill your processes. |
| pkg_chaos_monkey_homepage = https://github.com/dLuna/chaos_monkey |
| pkg_chaos_monkey_fetch = git |
| pkg_chaos_monkey_repo = https://github.com/dLuna/chaos_monkey |
| pkg_chaos_monkey_commit = master |
| |
| PACKAGES += check_node |
| pkg_check_node_name = check_node |
| pkg_check_node_description = Nagios Scripts for monitoring Riak |
| pkg_check_node_homepage = https://github.com/basho-labs/riak_nagios |
| pkg_check_node_fetch = git |
| pkg_check_node_repo = https://github.com/basho-labs/riak_nagios |
| pkg_check_node_commit = master |
| |
| PACKAGES += chronos |
| pkg_chronos_name = chronos |
| pkg_chronos_description = Timer module for Erlang that makes it easy to abstract time out of the tests. |
| pkg_chronos_homepage = https://github.com/lehoff/chronos |
| pkg_chronos_fetch = git |
| pkg_chronos_repo = https://github.com/lehoff/chronos |
| pkg_chronos_commit = master |
| |
| PACKAGES += chumak |
| pkg_chumak_name = chumak |
| pkg_chumak_description = Pure Erlang implementation of ZeroMQ Message Transport Protocol. |
| pkg_chumak_homepage = http://choven.ca |
| pkg_chumak_fetch = git |
| pkg_chumak_repo = https://github.com/chovencorp/chumak |
| pkg_chumak_commit = master |
| |
| PACKAGES += cl |
| pkg_cl_name = cl |
| pkg_cl_description = OpenCL binding for Erlang |
| pkg_cl_homepage = https://github.com/tonyrog/cl |
| pkg_cl_fetch = git |
| pkg_cl_repo = https://github.com/tonyrog/cl |
| pkg_cl_commit = master |
| |
| PACKAGES += clique |
| pkg_clique_name = clique |
| pkg_clique_description = CLI Framework for Erlang |
| pkg_clique_homepage = https://github.com/basho/clique |
| pkg_clique_fetch = git |
| pkg_clique_repo = https://github.com/basho/clique |
| pkg_clique_commit = develop |
| |
| PACKAGES += cloudi_core |
| pkg_cloudi_core_name = cloudi_core |
| pkg_cloudi_core_description = CloudI internal service runtime |
| pkg_cloudi_core_homepage = http://cloudi.org/ |
| pkg_cloudi_core_fetch = git |
| pkg_cloudi_core_repo = https://github.com/CloudI/cloudi_core |
| pkg_cloudi_core_commit = master |
| |
| PACKAGES += cloudi_service_api_requests |
| pkg_cloudi_service_api_requests_name = cloudi_service_api_requests |
| pkg_cloudi_service_api_requests_description = CloudI Service API requests (JSON-RPC/Erlang-term support) |
| pkg_cloudi_service_api_requests_homepage = http://cloudi.org/ |
| pkg_cloudi_service_api_requests_fetch = git |
| pkg_cloudi_service_api_requests_repo = https://github.com/CloudI/cloudi_service_api_requests |
| pkg_cloudi_service_api_requests_commit = master |
| |
| PACKAGES += cloudi_service_db_mysql |
| pkg_cloudi_service_db_mysql_name = cloudi_service_db_mysql |
| pkg_cloudi_service_db_mysql_description = MySQL CloudI Service |
| pkg_cloudi_service_db_mysql_homepage = http://cloudi.org/ |
| pkg_cloudi_service_db_mysql_fetch = git |
| pkg_cloudi_service_db_mysql_repo = https://github.com/CloudI/cloudi_service_db_mysql |
| pkg_cloudi_service_db_mysql_commit = master |
| |
| PACKAGES += cloudi_service_db_pgsql |
| pkg_cloudi_service_db_pgsql_name = cloudi_service_db_pgsql |
| pkg_cloudi_service_db_pgsql_description = PostgreSQL CloudI Service |
| pkg_cloudi_service_db_pgsql_homepage = http://cloudi.org/ |
| pkg_cloudi_service_db_pgsql_fetch = git |
| pkg_cloudi_service_db_pgsql_repo = https://github.com/CloudI/cloudi_service_db_pgsql |
| pkg_cloudi_service_db_pgsql_commit = master |
| |
| PACKAGES += cloudi_service_filesystem |
| pkg_cloudi_service_filesystem_name = cloudi_service_filesystem |
| pkg_cloudi_service_filesystem_description = Filesystem CloudI Service |
| pkg_cloudi_service_filesystem_homepage = http://cloudi.org/ |
| pkg_cloudi_service_filesystem_fetch = git |
| pkg_cloudi_service_filesystem_repo = https://github.com/CloudI/cloudi_service_filesystem |
| pkg_cloudi_service_filesystem_commit = master |
| |
| PACKAGES += cloudi_service_http_client |
| pkg_cloudi_service_http_client_name = cloudi_service_http_client |
| pkg_cloudi_service_http_client_description = HTTP client CloudI Service |
| pkg_cloudi_service_http_client_homepage = http://cloudi.org/ |
| pkg_cloudi_service_http_client_fetch = git |
| pkg_cloudi_service_http_client_repo = https://github.com/CloudI/cloudi_service_http_client |
| pkg_cloudi_service_http_client_commit = master |
| |
| PACKAGES += cloudi_service_http_cowboy |
| pkg_cloudi_service_http_cowboy_name = cloudi_service_http_cowboy |
| pkg_cloudi_service_http_cowboy_description = cowboy HTTP/HTTPS CloudI Service |
| pkg_cloudi_service_http_cowboy_homepage = http://cloudi.org/ |
| pkg_cloudi_service_http_cowboy_fetch = git |
| pkg_cloudi_service_http_cowboy_repo = https://github.com/CloudI/cloudi_service_http_cowboy |
| pkg_cloudi_service_http_cowboy_commit = master |
| |
| PACKAGES += cloudi_service_http_elli |
| pkg_cloudi_service_http_elli_name = cloudi_service_http_elli |
| pkg_cloudi_service_http_elli_description = elli HTTP CloudI Service |
| pkg_cloudi_service_http_elli_homepage = http://cloudi.org/ |
| pkg_cloudi_service_http_elli_fetch = git |
| pkg_cloudi_service_http_elli_repo = https://github.com/CloudI/cloudi_service_http_elli |
| pkg_cloudi_service_http_elli_commit = master |
| |
| PACKAGES += cloudi_service_map_reduce |
| pkg_cloudi_service_map_reduce_name = cloudi_service_map_reduce |
| pkg_cloudi_service_map_reduce_description = Map/Reduce CloudI Service |
| pkg_cloudi_service_map_reduce_homepage = http://cloudi.org/ |
| pkg_cloudi_service_map_reduce_fetch = git |
| pkg_cloudi_service_map_reduce_repo = https://github.com/CloudI/cloudi_service_map_reduce |
| pkg_cloudi_service_map_reduce_commit = master |
| |
| PACKAGES += cloudi_service_oauth1 |
| pkg_cloudi_service_oauth1_name = cloudi_service_oauth1 |
| pkg_cloudi_service_oauth1_description = OAuth v1.0 CloudI Service |
| pkg_cloudi_service_oauth1_homepage = http://cloudi.org/ |
| pkg_cloudi_service_oauth1_fetch = git |
| pkg_cloudi_service_oauth1_repo = https://github.com/CloudI/cloudi_service_oauth1 |
| pkg_cloudi_service_oauth1_commit = master |
| |
| PACKAGES += cloudi_service_queue |
| pkg_cloudi_service_queue_name = cloudi_service_queue |
| pkg_cloudi_service_queue_description = Persistent Queue Service |
| pkg_cloudi_service_queue_homepage = http://cloudi.org/ |
| pkg_cloudi_service_queue_fetch = git |
| pkg_cloudi_service_queue_repo = https://github.com/CloudI/cloudi_service_queue |
| pkg_cloudi_service_queue_commit = master |
| |
| PACKAGES += cloudi_service_quorum |
| pkg_cloudi_service_quorum_name = cloudi_service_quorum |
| pkg_cloudi_service_quorum_description = CloudI Quorum Service |
| pkg_cloudi_service_quorum_homepage = http://cloudi.org/ |
| pkg_cloudi_service_quorum_fetch = git |
| pkg_cloudi_service_quorum_repo = https://github.com/CloudI/cloudi_service_quorum |
| pkg_cloudi_service_quorum_commit = master |
| |
| PACKAGES += cloudi_service_router |
| pkg_cloudi_service_router_name = cloudi_service_router |
| pkg_cloudi_service_router_description = CloudI Router Service |
| pkg_cloudi_service_router_homepage = http://cloudi.org/ |
| pkg_cloudi_service_router_fetch = git |
| pkg_cloudi_service_router_repo = https://github.com/CloudI/cloudi_service_router |
| pkg_cloudi_service_router_commit = master |
| |
| PACKAGES += cloudi_service_tcp |
| pkg_cloudi_service_tcp_name = cloudi_service_tcp |
| pkg_cloudi_service_tcp_description = TCP CloudI Service |
| pkg_cloudi_service_tcp_homepage = http://cloudi.org/ |
| pkg_cloudi_service_tcp_fetch = git |
| pkg_cloudi_service_tcp_repo = https://github.com/CloudI/cloudi_service_tcp |
| pkg_cloudi_service_tcp_commit = master |
| |
| PACKAGES += cloudi_service_udp |
| pkg_cloudi_service_udp_name = cloudi_service_udp |
| pkg_cloudi_service_udp_description = UDP CloudI Service |
| pkg_cloudi_service_udp_homepage = http://cloudi.org/ |
| pkg_cloudi_service_udp_fetch = git |
| pkg_cloudi_service_udp_repo = https://github.com/CloudI/cloudi_service_udp |
| pkg_cloudi_service_udp_commit = master |
| |
| PACKAGES += cloudi_service_validate |
| pkg_cloudi_service_validate_name = cloudi_service_validate |
| pkg_cloudi_service_validate_description = CloudI Validate Service |
| pkg_cloudi_service_validate_homepage = http://cloudi.org/ |
| pkg_cloudi_service_validate_fetch = git |
| pkg_cloudi_service_validate_repo = https://github.com/CloudI/cloudi_service_validate |
| pkg_cloudi_service_validate_commit = master |
| |
| PACKAGES += cloudi_service_zeromq |
| pkg_cloudi_service_zeromq_name = cloudi_service_zeromq |
| pkg_cloudi_service_zeromq_description = ZeroMQ CloudI Service |
| pkg_cloudi_service_zeromq_homepage = http://cloudi.org/ |
| pkg_cloudi_service_zeromq_fetch = git |
| pkg_cloudi_service_zeromq_repo = https://github.com/CloudI/cloudi_service_zeromq |
| pkg_cloudi_service_zeromq_commit = master |
| |
| PACKAGES += cluster_info |
| pkg_cluster_info_name = cluster_info |
| pkg_cluster_info_description = Fork of Hibari's nifty cluster_info OTP app |
| pkg_cluster_info_homepage = https://github.com/basho/cluster_info |
| pkg_cluster_info_fetch = git |
| pkg_cluster_info_repo = https://github.com/basho/cluster_info |
| pkg_cluster_info_commit = master |
| |
| PACKAGES += color |
| pkg_color_name = color |
| pkg_color_description = ANSI colors for your Erlang |
| pkg_color_homepage = https://github.com/julianduque/erlang-color |
| pkg_color_fetch = git |
| pkg_color_repo = https://github.com/julianduque/erlang-color |
| pkg_color_commit = master |
| |
| PACKAGES += confetti |
| pkg_confetti_name = confetti |
| pkg_confetti_description = Erlang configuration provider / application:get_env/2 on steroids |
| pkg_confetti_homepage = https://github.com/jtendo/confetti |
| pkg_confetti_fetch = git |
| pkg_confetti_repo = https://github.com/jtendo/confetti |
| pkg_confetti_commit = master |
| |
| PACKAGES += couchbeam |
| pkg_couchbeam_name = couchbeam |
| pkg_couchbeam_description = Apache CouchDB client in Erlang |
| pkg_couchbeam_homepage = https://github.com/benoitc/couchbeam |
| pkg_couchbeam_fetch = git |
| pkg_couchbeam_repo = https://github.com/benoitc/couchbeam |
| pkg_couchbeam_commit = master |
| |
| PACKAGES += covertool |
| pkg_covertool_name = covertool |
| pkg_covertool_description = Tool to convert Erlang cover data files into Cobertura XML reports |
| pkg_covertool_homepage = https://github.com/idubrov/covertool |
| pkg_covertool_fetch = git |
| pkg_covertool_repo = https://github.com/idubrov/covertool |
| pkg_covertool_commit = master |
| |
| PACKAGES += cowboy |
| pkg_cowboy_name = cowboy |
| pkg_cowboy_description = Small, fast and modular HTTP server. |
| pkg_cowboy_homepage = http://ninenines.eu |
| pkg_cowboy_fetch = git |
| pkg_cowboy_repo = https://github.com/ninenines/cowboy |
| pkg_cowboy_commit = 1.0.4 |
| |
| PACKAGES += cowdb |
| pkg_cowdb_name = cowdb |
| pkg_cowdb_description = Pure Key/Value database library for Erlang Applications |
| pkg_cowdb_homepage = https://github.com/refuge/cowdb |
| pkg_cowdb_fetch = git |
| pkg_cowdb_repo = https://github.com/refuge/cowdb |
| pkg_cowdb_commit = master |
| |
| PACKAGES += cowlib |
| pkg_cowlib_name = cowlib |
| pkg_cowlib_description = Support library for manipulating Web protocols. |
| pkg_cowlib_homepage = http://ninenines.eu |
| pkg_cowlib_fetch = git |
| pkg_cowlib_repo = https://github.com/ninenines/cowlib |
| pkg_cowlib_commit = 1.0.2 |
| |
| PACKAGES += cpg |
| pkg_cpg_name = cpg |
| pkg_cpg_description = CloudI Process Groups |
| pkg_cpg_homepage = https://github.com/okeuday/cpg |
| pkg_cpg_fetch = git |
| pkg_cpg_repo = https://github.com/okeuday/cpg |
| pkg_cpg_commit = master |
| |
| PACKAGES += cqerl |
| pkg_cqerl_name = cqerl |
| pkg_cqerl_description = Native Erlang CQL client for Cassandra |
| pkg_cqerl_homepage = https://matehat.github.io/cqerl/ |
| pkg_cqerl_fetch = git |
| pkg_cqerl_repo = https://github.com/matehat/cqerl |
| pkg_cqerl_commit = master |
| |
| PACKAGES += cr |
| pkg_cr_name = cr |
| pkg_cr_description = Chain Replication |
| pkg_cr_homepage = https://synrc.com/apps/cr/doc/cr.htm |
| pkg_cr_fetch = git |
| pkg_cr_repo = https://github.com/spawnproc/cr |
| pkg_cr_commit = master |
| |
| PACKAGES += cuttlefish |
| pkg_cuttlefish_name = cuttlefish |
| pkg_cuttlefish_description = cuttlefish configuration abstraction |
| pkg_cuttlefish_homepage = https://github.com/Kyorai/cuttlefish |
| pkg_cuttlefish_fetch = git |
| pkg_cuttlefish_repo = https://github.com/Kyorai/cuttlefish |
| pkg_cuttlefish_commit = main |
| |
| PACKAGES += damocles |
| pkg_damocles_name = damocles |
| pkg_damocles_description = Erlang library for generating adversarial network conditions for QAing distributed applications/systems on a single Linux box. |
| pkg_damocles_homepage = https://github.com/lostcolony/damocles |
| pkg_damocles_fetch = git |
| pkg_damocles_repo = https://github.com/lostcolony/damocles |
| pkg_damocles_commit = master |
| |
| PACKAGES += debbie |
| pkg_debbie_name = debbie |
| pkg_debbie_description = .DEB Built In Erlang |
| pkg_debbie_homepage = https://github.com/crownedgrouse/debbie |
| pkg_debbie_fetch = git |
| pkg_debbie_repo = https://github.com/crownedgrouse/debbie |
| pkg_debbie_commit = master |
| |
| PACKAGES += decimal |
| pkg_decimal_name = decimal |
| pkg_decimal_description = An Erlang decimal arithmetic library |
| pkg_decimal_homepage = https://github.com/egobrain/decimal |
| pkg_decimal_fetch = git |
| pkg_decimal_repo = https://github.com/egobrain/decimal |
| pkg_decimal_commit = master |
| |
| PACKAGES += detergent |
| pkg_detergent_name = detergent |
| pkg_detergent_description = An emulsifying Erlang SOAP library |
| pkg_detergent_homepage = https://github.com/devinus/detergent |
| pkg_detergent_fetch = git |
| pkg_detergent_repo = https://github.com/devinus/detergent |
| pkg_detergent_commit = master |
| |
| PACKAGES += dh_date |
| pkg_dh_date_name = dh_date |
| pkg_dh_date_description = Date formatting / parsing library for erlang |
| pkg_dh_date_homepage = https://github.com/daleharvey/dh_date |
| pkg_dh_date_fetch = git |
| pkg_dh_date_repo = https://github.com/daleharvey/dh_date |
| pkg_dh_date_commit = master |
| |
| PACKAGES += dirbusterl |
| pkg_dirbusterl_name = dirbusterl |
| pkg_dirbusterl_description = DirBuster successor in Erlang |
| pkg_dirbusterl_homepage = https://github.com/silentsignal/DirBustErl |
| pkg_dirbusterl_fetch = git |
| pkg_dirbusterl_repo = https://github.com/silentsignal/DirBustErl |
| pkg_dirbusterl_commit = master |
| |
| PACKAGES += dispcount |
| pkg_dispcount_name = dispcount |
| pkg_dispcount_description = Erlang task dispatcher based on ETS counters. |
| pkg_dispcount_homepage = https://github.com/ferd/dispcount |
| pkg_dispcount_fetch = git |
| pkg_dispcount_repo = https://github.com/ferd/dispcount |
| pkg_dispcount_commit = master |
| |
| PACKAGES += dlhttpc |
| pkg_dlhttpc_name = dlhttpc |
| pkg_dlhttpc_description = dispcount-based lhttpc fork for massive amounts of requests to limited endpoints |
| pkg_dlhttpc_homepage = https://github.com/ferd/dlhttpc |
| pkg_dlhttpc_fetch = git |
| pkg_dlhttpc_repo = https://github.com/ferd/dlhttpc |
| pkg_dlhttpc_commit = master |
| |
| PACKAGES += dns |
| pkg_dns_name = dns |
| pkg_dns_description = Erlang DNS library |
| pkg_dns_homepage = https://github.com/aetrion/dns_erlang |
| pkg_dns_fetch = git |
| pkg_dns_repo = https://github.com/aetrion/dns_erlang |
| pkg_dns_commit = main |
| |
| PACKAGES += dynamic_compile |
| pkg_dynamic_compile_name = dynamic_compile |
| pkg_dynamic_compile_description = compile and load erlang modules from string input |
| pkg_dynamic_compile_homepage = https://github.com/jkvor/dynamic_compile |
| pkg_dynamic_compile_fetch = git |
| pkg_dynamic_compile_repo = https://github.com/jkvor/dynamic_compile |
| pkg_dynamic_compile_commit = master |
| |
| PACKAGES += e2 |
| pkg_e2_name = e2 |
| pkg_e2_description = Library to simply writing correct OTP applications. |
| pkg_e2_homepage = http://e2project.org |
| pkg_e2_fetch = git |
| pkg_e2_repo = https://github.com/gar1t/e2 |
| pkg_e2_commit = master |
| |
| PACKAGES += eamf |
| pkg_eamf_name = eamf |
| pkg_eamf_description = eAMF provides Action Message Format (AMF) support for Erlang |
| pkg_eamf_homepage = https://github.com/mrinalwadhwa/eamf |
| pkg_eamf_fetch = git |
| pkg_eamf_repo = https://github.com/mrinalwadhwa/eamf |
| pkg_eamf_commit = master |
| |
| PACKAGES += eavro |
| pkg_eavro_name = eavro |
| pkg_eavro_description = Apache Avro encoder/decoder |
| pkg_eavro_homepage = https://github.com/SIfoxDevTeam/eavro |
| pkg_eavro_fetch = git |
| pkg_eavro_repo = https://github.com/SIfoxDevTeam/eavro |
| pkg_eavro_commit = master |
| |
| PACKAGES += ecapnp |
| pkg_ecapnp_name = ecapnp |
| pkg_ecapnp_description = Cap'n Proto library for Erlang |
| pkg_ecapnp_homepage = https://github.com/kaos/ecapnp |
| pkg_ecapnp_fetch = git |
| pkg_ecapnp_repo = https://github.com/kaos/ecapnp |
| pkg_ecapnp_commit = master |
| |
| PACKAGES += econfig |
| pkg_econfig_name = econfig |
| pkg_econfig_description = simple Erlang config handler using INI files |
| pkg_econfig_homepage = https://github.com/benoitc/econfig |
| pkg_econfig_fetch = git |
| pkg_econfig_repo = https://github.com/benoitc/econfig |
| pkg_econfig_commit = master |
| |
| PACKAGES += edate |
| pkg_edate_name = edate |
| pkg_edate_description = date manipulation library for erlang |
| pkg_edate_homepage = https://github.com/dweldon/edate |
| pkg_edate_fetch = git |
| pkg_edate_repo = https://github.com/dweldon/edate |
| pkg_edate_commit = master |
| |
| PACKAGES += edgar |
| pkg_edgar_name = edgar |
| pkg_edgar_description = Erlang Does GNU AR |
| pkg_edgar_homepage = https://github.com/crownedgrouse/edgar |
| pkg_edgar_fetch = git |
| pkg_edgar_repo = https://github.com/crownedgrouse/edgar |
| pkg_edgar_commit = master |
| |
| PACKAGES += edns |
| pkg_edns_name = edns |
| pkg_edns_description = Erlang/OTP DNS server |
| pkg_edns_homepage = https://github.com/hcvst/erlang-dns |
| pkg_edns_fetch = git |
| pkg_edns_repo = https://github.com/hcvst/erlang-dns |
| pkg_edns_commit = master |
| |
| PACKAGES += edown |
| pkg_edown_name = edown |
| pkg_edown_description = EDoc extension for generating Github-flavored Markdown |
| pkg_edown_homepage = https://github.com/uwiger/edown |
| pkg_edown_fetch = git |
| pkg_edown_repo = https://github.com/uwiger/edown |
| pkg_edown_commit = master |
| |
| PACKAGES += eep |
| pkg_eep_name = eep |
| pkg_eep_description = Erlang Easy Profiling (eep) application provides a way to analyze application performance and call hierarchy |
| pkg_eep_homepage = https://github.com/virtan/eep |
| pkg_eep_fetch = git |
| pkg_eep_repo = https://github.com/virtan/eep |
| pkg_eep_commit = master |
| |
| PACKAGES += eep_app |
| pkg_eep_app_name = eep_app |
| pkg_eep_app_description = Embedded Event Processing |
| pkg_eep_app_homepage = https://github.com/darach/eep-erl |
| pkg_eep_app_fetch = git |
| pkg_eep_app_repo = https://github.com/darach/eep-erl |
| pkg_eep_app_commit = master |
| |
| PACKAGES += efene |
| pkg_efene_name = efene |
| pkg_efene_description = Alternative syntax for the Erlang Programming Language focusing on simplicity, ease of use and programmer UX |
| pkg_efene_homepage = https://github.com/efene/efene |
| pkg_efene_fetch = git |
| pkg_efene_repo = https://github.com/efene/efene |
| pkg_efene_commit = master |
| |
| PACKAGES += egeoip |
| pkg_egeoip_name = egeoip |
| pkg_egeoip_description = Erlang IP Geolocation module, currently supporting the MaxMind GeoLite City Database. |
| pkg_egeoip_homepage = https://github.com/mochi/egeoip |
| pkg_egeoip_fetch = git |
| pkg_egeoip_repo = https://github.com/mochi/egeoip |
| pkg_egeoip_commit = master |
| |
| PACKAGES += ehsa |
| pkg_ehsa_name = ehsa |
| pkg_ehsa_description = Erlang HTTP server basic and digest authentication modules |
| pkg_ehsa_homepage = https://github.com/a12n/ehsa |
| pkg_ehsa_fetch = git |
| pkg_ehsa_repo = https://github.com/a12n/ehsa |
| pkg_ehsa_commit = master |
| |
| PACKAGES += ej |
| pkg_ej_name = ej |
| pkg_ej_description = Helper module for working with Erlang terms representing JSON |
| pkg_ej_homepage = https://github.com/seth/ej |
| pkg_ej_fetch = git |
| pkg_ej_repo = https://github.com/seth/ej |
| pkg_ej_commit = master |
| |
| PACKAGES += ejabberd |
| pkg_ejabberd_name = ejabberd |
| pkg_ejabberd_description = Robust, ubiquitous and massively scalable Jabber / XMPP Instant Messaging platform |
| pkg_ejabberd_homepage = https://github.com/processone/ejabberd |
| pkg_ejabberd_fetch = git |
| pkg_ejabberd_repo = https://github.com/processone/ejabberd |
| pkg_ejabberd_commit = master |
| |
| PACKAGES += ejwt |
| pkg_ejwt_name = ejwt |
| pkg_ejwt_description = erlang library for JSON Web Token |
| pkg_ejwt_homepage = https://github.com/artefactop/ejwt |
| pkg_ejwt_fetch = git |
| pkg_ejwt_repo = https://github.com/artefactop/ejwt |
| pkg_ejwt_commit = master |
| |
| PACKAGES += ekaf |
| pkg_ekaf_name = ekaf |
| pkg_ekaf_description = A minimal, high-performance Kafka client in Erlang. |
| pkg_ekaf_homepage = https://github.com/helpshift/ekaf |
| pkg_ekaf_fetch = git |
| pkg_ekaf_repo = https://github.com/helpshift/ekaf |
| pkg_ekaf_commit = master |
| |
| PACKAGES += elarm |
| pkg_elarm_name = elarm |
| pkg_elarm_description = Alarm Manager for Erlang. |
| pkg_elarm_homepage = https://github.com/esl/elarm |
| pkg_elarm_fetch = git |
| pkg_elarm_repo = https://github.com/esl/elarm |
| pkg_elarm_commit = master |
| |
| PACKAGES += eleveldb |
| pkg_eleveldb_name = eleveldb |
| pkg_eleveldb_description = Erlang LevelDB API |
| pkg_eleveldb_homepage = https://github.com/basho/eleveldb |
| pkg_eleveldb_fetch = git |
| pkg_eleveldb_repo = https://github.com/basho/eleveldb |
| pkg_eleveldb_commit = develop |
| |
| PACKAGES += elixir |
| pkg_elixir_name = elixir |
| pkg_elixir_description = Elixir is a dynamic, functional language designed for building scalable and maintainable applications |
| pkg_elixir_homepage = https://elixir-lang.org/ |
| pkg_elixir_fetch = git |
| pkg_elixir_repo = https://github.com/elixir-lang/elixir |
| pkg_elixir_commit = main |
| |
| PACKAGES += elli |
| pkg_elli_name = elli |
| pkg_elli_description = Simple, robust and performant Erlang web server |
| pkg_elli_homepage = https://github.com/elli-lib/elli |
| pkg_elli_fetch = git |
| pkg_elli_repo = https://github.com/elli-lib/elli |
| pkg_elli_commit = main |
| |
| PACKAGES += elvis |
| pkg_elvis_name = elvis |
| pkg_elvis_description = Erlang Style Reviewer |
| pkg_elvis_homepage = https://github.com/inaka/elvis |
| pkg_elvis_fetch = git |
| pkg_elvis_repo = https://github.com/inaka/elvis |
| pkg_elvis_commit = master |
| |
| PACKAGES += emagick |
| pkg_emagick_name = emagick |
| pkg_emagick_description = Wrapper for Graphics/ImageMagick command line tool. |
| pkg_emagick_homepage = https://github.com/kivra/emagick |
| pkg_emagick_fetch = git |
| pkg_emagick_repo = https://github.com/kivra/emagick |
| pkg_emagick_commit = master |
| |
| PACKAGES += enm |
| pkg_enm_name = enm |
| pkg_enm_description = Erlang driver for nanomsg |
| pkg_enm_homepage = https://github.com/basho/enm |
| pkg_enm_fetch = git |
| pkg_enm_repo = https://github.com/basho/enm |
| pkg_enm_commit = master |
| |
| PACKAGES += entop |
| pkg_entop_name = entop |
| pkg_entop_description = A top-like tool for monitoring an Erlang node |
| pkg_entop_homepage = https://github.com/mazenharake/entop |
| pkg_entop_fetch = git |
| pkg_entop_repo = https://github.com/mazenharake/entop |
| pkg_entop_commit = master |
| |
| PACKAGES += epcap |
| pkg_epcap_name = epcap |
| pkg_epcap_description = Erlang packet capture interface using pcap |
| pkg_epcap_homepage = https://github.com/msantos/epcap |
| pkg_epcap_fetch = git |
| pkg_epcap_repo = https://github.com/msantos/epcap |
| pkg_epcap_commit = master |
| |
| PACKAGES += eper |
| pkg_eper_name = eper |
| pkg_eper_description = Erlang performance and debugging tools. |
| pkg_eper_homepage = https://github.com/massemanet/eper |
| pkg_eper_fetch = git |
| pkg_eper_repo = https://github.com/massemanet/eper |
| pkg_eper_commit = master |
| |
| PACKAGES += epgsql |
| pkg_epgsql_name = epgsql |
| pkg_epgsql_description = Erlang PostgreSQL client library. |
| pkg_epgsql_homepage = https://github.com/epgsql/epgsql |
| pkg_epgsql_fetch = git |
| pkg_epgsql_repo = https://github.com/epgsql/epgsql |
| pkg_epgsql_commit = master |
| |
| PACKAGES += episcina |
| pkg_episcina_name = episcina |
| pkg_episcina_description = A simple non intrusive resource pool for connections |
| pkg_episcina_homepage = https://github.com/erlware/episcina |
| pkg_episcina_fetch = git |
| pkg_episcina_repo = https://github.com/erlware/episcina |
| pkg_episcina_commit = master |
| |
| PACKAGES += eplot |
| pkg_eplot_name = eplot |
| pkg_eplot_description = A plot engine written in erlang. |
| pkg_eplot_homepage = https://github.com/psyeugenic/eplot |
| pkg_eplot_fetch = git |
| pkg_eplot_repo = https://github.com/psyeugenic/eplot |
| pkg_eplot_commit = master |
| |
| PACKAGES += epocxy |
| pkg_epocxy_name = epocxy |
| pkg_epocxy_description = Erlang Patterns of Concurrency |
| pkg_epocxy_homepage = https://github.com/duomark/epocxy |
| pkg_epocxy_fetch = git |
| pkg_epocxy_repo = https://github.com/duomark/epocxy |
| pkg_epocxy_commit = master |
| |
| PACKAGES += epubnub |
| pkg_epubnub_name = epubnub |
| pkg_epubnub_description = Erlang PubNub API |
| pkg_epubnub_homepage = https://github.com/tsloughter/epubnub |
| pkg_epubnub_fetch = git |
| pkg_epubnub_repo = https://github.com/tsloughter/epubnub |
| pkg_epubnub_commit = master |
| |
| PACKAGES += eqm |
| pkg_eqm_name = eqm |
| pkg_eqm_description = Erlang pub sub with supply-demand channels |
| pkg_eqm_homepage = https://github.com/loucash/eqm |
| pkg_eqm_fetch = git |
| pkg_eqm_repo = https://github.com/loucash/eqm |
| pkg_eqm_commit = master |
| |
| PACKAGES += eredis |
| pkg_eredis_name = eredis |
| pkg_eredis_description = Erlang Redis client |
| pkg_eredis_homepage = https://github.com/wooga/eredis |
| pkg_eredis_fetch = git |
| pkg_eredis_repo = https://github.com/wooga/eredis |
| pkg_eredis_commit = master |
| |
| PACKAGES += erl_streams |
| pkg_erl_streams_name = erl_streams |
| pkg_erl_streams_description = Streams in Erlang |
| pkg_erl_streams_homepage = https://github.com/epappas/erl_streams |
| pkg_erl_streams_fetch = git |
| pkg_erl_streams_repo = https://github.com/epappas/erl_streams |
| pkg_erl_streams_commit = master |
| |
| PACKAGES += erlang_localtime |
| pkg_erlang_localtime_name = erlang_localtime |
| pkg_erlang_localtime_description = Erlang library for conversion from one local time to another |
| pkg_erlang_localtime_homepage = https://github.com/dmitryme/erlang_localtime |
| pkg_erlang_localtime_fetch = git |
| pkg_erlang_localtime_repo = https://github.com/dmitryme/erlang_localtime |
| pkg_erlang_localtime_commit = master |
| |
| PACKAGES += erlang_smtp |
| pkg_erlang_smtp_name = erlang_smtp |
| pkg_erlang_smtp_description = Erlang SMTP and POP3 server code. |
| pkg_erlang_smtp_homepage = https://github.com/tonyg/erlang-smtp |
| pkg_erlang_smtp_fetch = git |
| pkg_erlang_smtp_repo = https://github.com/tonyg/erlang-smtp |
| pkg_erlang_smtp_commit = master |
| |
| PACKAGES += erlang_term |
| pkg_erlang_term_name = erlang_term |
| pkg_erlang_term_description = Erlang Term Info |
| pkg_erlang_term_homepage = https://github.com/okeuday/erlang_term |
| pkg_erlang_term_fetch = git |
| pkg_erlang_term_repo = https://github.com/okeuday/erlang_term |
| pkg_erlang_term_commit = master |
| |
| PACKAGES += erlastic_search |
| pkg_erlastic_search_name = erlastic_search |
| pkg_erlastic_search_description = An Erlang app for communicating with Elastic Search's rest interface. |
| pkg_erlastic_search_homepage = https://github.com/tsloughter/erlastic_search |
| pkg_erlastic_search_fetch = git |
| pkg_erlastic_search_repo = https://github.com/tsloughter/erlastic_search |
| pkg_erlastic_search_commit = master |
| |
| PACKAGES += erlbrake |
| pkg_erlbrake_name = erlbrake |
| pkg_erlbrake_description = Erlang Airbrake notification client |
| pkg_erlbrake_homepage = https://github.com/kenpratt/erlbrake |
| pkg_erlbrake_fetch = git |
| pkg_erlbrake_repo = https://github.com/kenpratt/erlbrake |
| pkg_erlbrake_commit = master |
| |
| PACKAGES += erlcloud |
| pkg_erlcloud_name = erlcloud |
| pkg_erlcloud_description = Cloud Computing library for erlang (Amazon EC2, S3, SQS, SimpleDB, Mechanical Turk, ELB) |
| pkg_erlcloud_homepage = https://github.com/gleber/erlcloud |
| pkg_erlcloud_fetch = git |
| pkg_erlcloud_repo = https://github.com/gleber/erlcloud |
| pkg_erlcloud_commit = master |
| |
| PACKAGES += erlcron |
| pkg_erlcron_name = erlcron |
| pkg_erlcron_description = Erlang cronish system |
| pkg_erlcron_homepage = https://github.com/erlware/erlcron |
| pkg_erlcron_fetch = git |
| pkg_erlcron_repo = https://github.com/erlware/erlcron |
| pkg_erlcron_commit = master |
| |
| PACKAGES += erldb |
| pkg_erldb_name = erldb |
| pkg_erldb_description = ORM (Object-relational mapping) application implemented in Erlang |
| pkg_erldb_homepage = http://erldb.org |
| pkg_erldb_fetch = git |
| pkg_erldb_repo = https://github.com/erldb/erldb |
| pkg_erldb_commit = master |
| |
| PACKAGES += erldis |
| pkg_erldis_name = erldis |
| pkg_erldis_description = redis erlang client library |
| pkg_erldis_homepage = https://github.com/cstar/erldis |
| pkg_erldis_fetch = git |
| pkg_erldis_repo = https://github.com/cstar/erldis |
| pkg_erldis_commit = master |
| |
| PACKAGES += erldns |
| pkg_erldns_name = erldns |
| pkg_erldns_description = DNS server, in erlang. |
| pkg_erldns_homepage = https://github.com/aetrion/erl-dns |
| pkg_erldns_fetch = git |
| pkg_erldns_repo = https://github.com/aetrion/erl-dns |
| pkg_erldns_commit = main |
| |
| PACKAGES += erldocker |
| pkg_erldocker_name = erldocker |
| pkg_erldocker_description = Docker Remote API client for Erlang |
| pkg_erldocker_homepage = https://github.com/proger/erldocker |
| pkg_erldocker_fetch = git |
| pkg_erldocker_repo = https://github.com/proger/erldocker |
| pkg_erldocker_commit = master |
| |
| PACKAGES += erlfsmon |
| pkg_erlfsmon_name = erlfsmon |
| pkg_erlfsmon_description = Erlang filesystem event watcher for Linux and OSX |
| pkg_erlfsmon_homepage = https://github.com/proger/erlfsmon |
| pkg_erlfsmon_fetch = git |
| pkg_erlfsmon_repo = https://github.com/proger/erlfsmon |
| pkg_erlfsmon_commit = master |
| |
| PACKAGES += erlgit |
| pkg_erlgit_name = erlgit |
| pkg_erlgit_description = Erlang convenience wrapper around git executable |
| pkg_erlgit_homepage = https://github.com/gleber/erlgit |
| pkg_erlgit_fetch = git |
| pkg_erlgit_repo = https://github.com/gleber/erlgit |
| pkg_erlgit_commit = master |
| |
| PACKAGES += erlguten |
| pkg_erlguten_name = erlguten |
| pkg_erlguten_description = ErlGuten is a system for high-quality typesetting, written purely in Erlang. |
| pkg_erlguten_homepage = https://github.com/richcarl/erlguten |
| pkg_erlguten_fetch = git |
| pkg_erlguten_repo = https://github.com/richcarl/erlguten |
| pkg_erlguten_commit = master |
| |
| PACKAGES += erlmc |
| pkg_erlmc_name = erlmc |
| pkg_erlmc_description = Erlang memcached binary protocol client |
| pkg_erlmc_homepage = https://github.com/jkvor/erlmc |
| pkg_erlmc_fetch = git |
| pkg_erlmc_repo = https://github.com/jkvor/erlmc |
| pkg_erlmc_commit = master |
| |
| PACKAGES += erlmongo |
| pkg_erlmongo_name = erlmongo |
| pkg_erlmongo_description = Record based Erlang driver for MongoDB with gridfs support |
| pkg_erlmongo_homepage = https://github.com/SergejJurecko/erlmongo |
| pkg_erlmongo_fetch = git |
| pkg_erlmongo_repo = https://github.com/SergejJurecko/erlmongo |
| pkg_erlmongo_commit = master |
| |
| PACKAGES += erlog |
| pkg_erlog_name = erlog |
| pkg_erlog_description = Prolog interpreter in and for Erlang |
| pkg_erlog_homepage = https://github.com/rvirding/erlog |
| pkg_erlog_fetch = git |
| pkg_erlog_repo = https://github.com/rvirding/erlog |
| pkg_erlog_commit = master |
| |
| PACKAGES += erlpass |
| pkg_erlpass_name = erlpass |
| pkg_erlpass_description = A library to handle password hashing and changing in a safe manner, independent from any kind of storage whatsoever. |
| pkg_erlpass_homepage = https://github.com/ferd/erlpass |
| pkg_erlpass_fetch = git |
| pkg_erlpass_repo = https://github.com/ferd/erlpass |
| pkg_erlpass_commit = master |
| |
| PACKAGES += erlsh |
| pkg_erlsh_name = erlsh |
| pkg_erlsh_description = Erlang shell tools |
| pkg_erlsh_homepage = https://github.com/proger/erlsh |
| pkg_erlsh_fetch = git |
| pkg_erlsh_repo = https://github.com/proger/erlsh |
| pkg_erlsh_commit = master |
| |
| PACKAGES += erlsha2 |
| pkg_erlsha2_name = erlsha2 |
| pkg_erlsha2_description = SHA-224, SHA-256, SHA-384, SHA-512 implemented in Erlang NIFs. |
| pkg_erlsha2_homepage = https://github.com/vinoski/erlsha2 |
| pkg_erlsha2_fetch = git |
| pkg_erlsha2_repo = https://github.com/vinoski/erlsha2 |
| pkg_erlsha2_commit = master |
| |
| PACKAGES += erlsom |
| pkg_erlsom_name = erlsom |
| pkg_erlsom_description = XML parser for Erlang |
| pkg_erlsom_homepage = https://github.com/willemdj/erlsom |
| pkg_erlsom_fetch = git |
| pkg_erlsom_repo = https://github.com/willemdj/erlsom |
| pkg_erlsom_commit = master |
| |
| PACKAGES += erlubi |
| pkg_erlubi_name = erlubi |
| pkg_erlubi_description = Ubigraph Erlang Client (and Process Visualizer) |
| pkg_erlubi_homepage = https://github.com/krestenkrab/erlubi |
| pkg_erlubi_fetch = git |
| pkg_erlubi_repo = https://github.com/krestenkrab/erlubi |
| pkg_erlubi_commit = master |
| |
| PACKAGES += erlvolt |
| pkg_erlvolt_name = erlvolt |
| pkg_erlvolt_description = VoltDB Erlang Client Driver |
| pkg_erlvolt_homepage = https://github.com/VoltDB/voltdb-client-erlang |
| pkg_erlvolt_fetch = git |
| pkg_erlvolt_repo = https://github.com/VoltDB/voltdb-client-erlang |
| pkg_erlvolt_commit = master |
| |
| PACKAGES += erlware_commons |
| pkg_erlware_commons_name = erlware_commons |
| pkg_erlware_commons_description = Erlware Commons is an Erlware project focused on all aspects of reusable Erlang components. |
| pkg_erlware_commons_homepage = https://github.com/erlware/erlware_commons |
| pkg_erlware_commons_fetch = git |
| pkg_erlware_commons_repo = https://github.com/erlware/erlware_commons |
| pkg_erlware_commons_commit = master |
| |
| PACKAGES += erlydtl |
| pkg_erlydtl_name = erlydtl |
| pkg_erlydtl_description = Django Template Language for Erlang. |
| pkg_erlydtl_homepage = https://github.com/erlydtl/erlydtl |
| pkg_erlydtl_fetch = git |
| pkg_erlydtl_repo = https://github.com/erlydtl/erlydtl |
| pkg_erlydtl_commit = master |
| |
| PACKAGES += errd |
| pkg_errd_name = errd |
| pkg_errd_description = Erlang RRDTool library |
| pkg_errd_homepage = https://github.com/archaelus/errd |
| pkg_errd_fetch = git |
| pkg_errd_repo = https://github.com/archaelus/errd |
| pkg_errd_commit = master |
| |
| PACKAGES += erserve |
| pkg_erserve_name = erserve |
| pkg_erserve_description = Erlang/Rserve communication interface |
| pkg_erserve_homepage = https://github.com/del/erserve |
| pkg_erserve_fetch = git |
| pkg_erserve_repo = https://github.com/del/erserve |
| pkg_erserve_commit = master |
| |
| PACKAGES += escalus |
| pkg_escalus_name = escalus |
| pkg_escalus_description = An XMPP client library in Erlang for conveniently testing XMPP servers |
| pkg_escalus_homepage = https://github.com/esl/escalus |
| pkg_escalus_fetch = git |
| pkg_escalus_repo = https://github.com/esl/escalus |
| pkg_escalus_commit = master |
| |
| PACKAGES += esh_mk |
| pkg_esh_mk_name = esh_mk |
| pkg_esh_mk_description = esh template engine plugin for erlang.mk |
| pkg_esh_mk_homepage = https://github.com/crownedgrouse/esh.mk |
| pkg_esh_mk_fetch = git |
| pkg_esh_mk_repo = https://github.com/crownedgrouse/esh.mk.git |
| pkg_esh_mk_commit = master |
| |
| PACKAGES += espec |
| pkg_espec_name = espec |
| pkg_espec_description = ESpec: Behaviour driven development framework for Erlang |
| pkg_espec_homepage = https://github.com/lucaspiller/espec |
| pkg_espec_fetch = git |
| pkg_espec_repo = https://github.com/lucaspiller/espec |
| pkg_espec_commit = master |
| |
| PACKAGES += estatsd |
| pkg_estatsd_name = estatsd |
| pkg_estatsd_description = Erlang stats aggregation app that periodically flushes data to graphite |
| pkg_estatsd_homepage = https://github.com/RJ/estatsd |
| pkg_estatsd_fetch = git |
| pkg_estatsd_repo = https://github.com/RJ/estatsd |
| pkg_estatsd_commit = master |
| |
| PACKAGES += etap |
| pkg_etap_name = etap |
| pkg_etap_description = etap is a simple erlang testing library that provides TAP compliant output. |
| pkg_etap_homepage = https://github.com/ngerakines/etap |
| pkg_etap_fetch = git |
| pkg_etap_repo = https://github.com/ngerakines/etap |
| pkg_etap_commit = master |
| |
| PACKAGES += etest |
| pkg_etest_name = etest |
| pkg_etest_description = A lightweight, convention over configuration test framework for Erlang |
| pkg_etest_homepage = https://github.com/wooga/etest |
| pkg_etest_fetch = git |
| pkg_etest_repo = https://github.com/wooga/etest |
| pkg_etest_commit = master |
| |
| PACKAGES += etest_http |
| pkg_etest_http_name = etest_http |
| pkg_etest_http_description = etest Assertions around HTTP (client-side) |
| pkg_etest_http_homepage = https://github.com/wooga/etest_http |
| pkg_etest_http_fetch = git |
| pkg_etest_http_repo = https://github.com/wooga/etest_http |
| pkg_etest_http_commit = master |
| |
| PACKAGES += etoml |
| pkg_etoml_name = etoml |
| pkg_etoml_description = TOML language erlang parser |
| pkg_etoml_homepage = https://github.com/kalta/etoml |
| pkg_etoml_fetch = git |
| pkg_etoml_repo = https://github.com/kalta/etoml |
| pkg_etoml_commit = master |
| |
| PACKAGES += eunit |
| pkg_eunit_name = eunit |
| pkg_eunit_description = The EUnit lightweight unit testing framework for Erlang - this is the canonical development repository. |
| pkg_eunit_homepage = https://github.com/richcarl/eunit |
| pkg_eunit_fetch = git |
| pkg_eunit_repo = https://github.com/richcarl/eunit |
| pkg_eunit_commit = master |
| |
| PACKAGES += eunit_formatters |
| pkg_eunit_formatters_name = eunit_formatters |
| pkg_eunit_formatters_description = Because eunit's output sucks. Let's make it better. |
| pkg_eunit_formatters_homepage = https://github.com/seancribbs/eunit_formatters |
| pkg_eunit_formatters_fetch = git |
| pkg_eunit_formatters_repo = https://github.com/seancribbs/eunit_formatters |
| pkg_eunit_formatters_commit = master |
| |
| PACKAGES += euthanasia |
| pkg_euthanasia_name = euthanasia |
| pkg_euthanasia_description = Merciful killer for your Erlang processes |
| pkg_euthanasia_homepage = https://github.com/doubleyou/euthanasia |
| pkg_euthanasia_fetch = git |
| pkg_euthanasia_repo = https://github.com/doubleyou/euthanasia |
| pkg_euthanasia_commit = master |
| |
| PACKAGES += evum |
| pkg_evum_name = evum |
| pkg_evum_description = Spawn Linux VMs as Erlang processes in the Erlang VM |
| pkg_evum_homepage = https://github.com/msantos/evum |
| pkg_evum_fetch = git |
| pkg_evum_repo = https://github.com/msantos/evum |
| pkg_evum_commit = master |
| |
| PACKAGES += exec |
| pkg_exec_name = erlexec |
| pkg_exec_description = Execute and control OS processes from Erlang/OTP. |
| pkg_exec_homepage = http://saleyn.github.com/erlexec |
| pkg_exec_fetch = git |
| pkg_exec_repo = https://github.com/saleyn/erlexec |
| pkg_exec_commit = master |
| |
| PACKAGES += exml |
| pkg_exml_name = exml |
| pkg_exml_description = XML parsing library in Erlang |
| pkg_exml_homepage = https://github.com/paulgray/exml |
| pkg_exml_fetch = git |
| pkg_exml_repo = https://github.com/paulgray/exml |
| pkg_exml_commit = master |
| |
| PACKAGES += exometer |
| pkg_exometer_name = exometer |
| pkg_exometer_description = Basic measurement objects and probe behavior |
| pkg_exometer_homepage = https://github.com/Feuerlabs/exometer |
| pkg_exometer_fetch = git |
| pkg_exometer_repo = https://github.com/Feuerlabs/exometer |
| pkg_exometer_commit = master |
| |
| PACKAGES += exs1024 |
| pkg_exs1024_name = exs1024 |
| pkg_exs1024_description = Xorshift1024star pseudo random number generator for Erlang. |
| pkg_exs1024_homepage = https://github.com/jj1bdx/exs1024 |
| pkg_exs1024_fetch = git |
| pkg_exs1024_repo = https://github.com/jj1bdx/exs1024 |
| pkg_exs1024_commit = master |
| |
| PACKAGES += exsplus116 |
| pkg_exsplus116_name = exsplus116 |
| pkg_exsplus116_description = Xorshift116plus for Erlang |
| pkg_exsplus116_homepage = https://github.com/jj1bdx/exsplus116 |
| pkg_exsplus116_fetch = git |
| pkg_exsplus116_repo = https://github.com/jj1bdx/exsplus116 |
| pkg_exsplus116_commit = master |
| |
| PACKAGES += ezmtp |
| pkg_ezmtp_name = ezmtp |
| pkg_ezmtp_description = ZMTP protocol in pure Erlang. |
| pkg_ezmtp_homepage = https://github.com/a13x/ezmtp |
| pkg_ezmtp_fetch = git |
| pkg_ezmtp_repo = https://github.com/a13x/ezmtp |
| pkg_ezmtp_commit = master |
| |
| PACKAGES += fast_disk_log |
| pkg_fast_disk_log_name = fast_disk_log |
| pkg_fast_disk_log_description = Pool-based asynchronous Erlang disk logger |
| pkg_fast_disk_log_homepage = https://github.com/lpgauth/fast_disk_log |
| pkg_fast_disk_log_fetch = git |
| pkg_fast_disk_log_repo = https://github.com/lpgauth/fast_disk_log |
| pkg_fast_disk_log_commit = master |
| |
| PACKAGES += feeder |
| pkg_feeder_name = feeder |
| pkg_feeder_description = Stream parse RSS and Atom formatted XML feeds. |
| pkg_feeder_homepage = https://github.com/michaelnisi/feeder |
| pkg_feeder_fetch = git |
| pkg_feeder_repo = https://github.com/michaelnisi/feeder |
| pkg_feeder_commit = master |
| |
| PACKAGES += find_crate |
| pkg_find_crate_name = find_crate |
| pkg_find_crate_description = Find Rust libs and exes in Erlang application priv directory |
| pkg_find_crate_homepage = https://github.com/goertzenator/find_crate |
| pkg_find_crate_fetch = git |
| pkg_find_crate_repo = https://github.com/goertzenator/find_crate |
| pkg_find_crate_commit = master |
| |
| PACKAGES += fix |
| pkg_fix_name = fix |
| pkg_fix_description = http://fixprotocol.org/ implementation. |
| pkg_fix_homepage = https://github.com/maxlapshin/fix |
| pkg_fix_fetch = git |
| pkg_fix_repo = https://github.com/maxlapshin/fix |
| pkg_fix_commit = master |
| |
| PACKAGES += flower |
| pkg_flower_name = flower |
| pkg_flower_description = FlowER - a Erlang OpenFlow development platform |
| pkg_flower_homepage = https://github.com/travelping/flower |
| pkg_flower_fetch = git |
| pkg_flower_repo = https://github.com/travelping/flower |
| pkg_flower_commit = master |
| |
| PACKAGES += fn |
| pkg_fn_name = fn |
| pkg_fn_description = Function utilities for Erlang |
| pkg_fn_homepage = https://github.com/reiddraper/fn |
| pkg_fn_fetch = git |
| pkg_fn_repo = https://github.com/reiddraper/fn |
| pkg_fn_commit = master |
| |
| PACKAGES += folsom |
| pkg_folsom_name = folsom |
| pkg_folsom_description = Expose Erlang Events and Metrics |
| pkg_folsom_homepage = https://github.com/boundary/folsom |
| pkg_folsom_fetch = git |
| pkg_folsom_repo = https://github.com/boundary/folsom |
| pkg_folsom_commit = master |
| |
| PACKAGES += folsom_cowboy |
| pkg_folsom_cowboy_name = folsom_cowboy |
| pkg_folsom_cowboy_description = A Cowboy based Folsom HTTP Wrapper. |
| pkg_folsom_cowboy_homepage = https://github.com/boundary/folsom_cowboy |
| pkg_folsom_cowboy_fetch = git |
| pkg_folsom_cowboy_repo = https://github.com/boundary/folsom_cowboy |
| pkg_folsom_cowboy_commit = master |
| |
| PACKAGES += fs |
| pkg_fs_name = fs |
| pkg_fs_description = Erlang FileSystem Listener |
| pkg_fs_homepage = https://github.com/synrc/fs |
| pkg_fs_fetch = git |
| pkg_fs_repo = https://github.com/synrc/fs |
| pkg_fs_commit = master |
| |
| PACKAGES += fuse |
| pkg_fuse_name = fuse |
| pkg_fuse_description = A Circuit Breaker for Erlang |
| pkg_fuse_homepage = https://github.com/jlouis/fuse |
| pkg_fuse_fetch = git |
| pkg_fuse_repo = https://github.com/jlouis/fuse |
| pkg_fuse_commit = master |
| |
| PACKAGES += gcm |
| pkg_gcm_name = gcm |
| pkg_gcm_description = An Erlang application for Google Cloud Messaging |
| pkg_gcm_homepage = https://github.com/pdincau/gcm-erlang |
| pkg_gcm_fetch = git |
| pkg_gcm_repo = https://github.com/pdincau/gcm-erlang |
| pkg_gcm_commit = master |
| |
| PACKAGES += gcprof |
| pkg_gcprof_name = gcprof |
| pkg_gcprof_description = Garbage Collection profiler for Erlang |
| pkg_gcprof_homepage = https://github.com/knutin/gcprof |
| pkg_gcprof_fetch = git |
| pkg_gcprof_repo = https://github.com/knutin/gcprof |
| pkg_gcprof_commit = master |
| |
| PACKAGES += geas |
| pkg_geas_name = geas |
| pkg_geas_description = Guess Erlang Application Scattering |
| pkg_geas_homepage = https://github.com/crownedgrouse/geas |
| pkg_geas_fetch = git |
| pkg_geas_repo = https://github.com/crownedgrouse/geas |
| pkg_geas_commit = master |
| |
| PACKAGES += geef |
| pkg_geef_name = geef |
| pkg_geef_description = Git NEEEEF (Erlang NIF) |
| pkg_geef_homepage = https://github.com/carlosmn/geef |
| pkg_geef_fetch = git |
| pkg_geef_repo = https://github.com/carlosmn/geef |
| pkg_geef_commit = master |
| |
| PACKAGES += gen_coap |
| pkg_gen_coap_name = gen_coap |
| pkg_gen_coap_description = Generic Erlang CoAP Client/Server |
| pkg_gen_coap_homepage = https://github.com/gotthardp/gen_coap |
| pkg_gen_coap_fetch = git |
| pkg_gen_coap_repo = https://github.com/gotthardp/gen_coap |
| pkg_gen_coap_commit = master |
| |
| PACKAGES += gen_cycle |
| pkg_gen_cycle_name = gen_cycle |
| pkg_gen_cycle_description = Simple, generic OTP behaviour for recurring tasks |
| pkg_gen_cycle_homepage = https://github.com/aerosol/gen_cycle |
| pkg_gen_cycle_fetch = git |
| pkg_gen_cycle_repo = https://github.com/aerosol/gen_cycle |
| pkg_gen_cycle_commit = develop |
| |
| PACKAGES += gen_icmp |
| pkg_gen_icmp_name = gen_icmp |
| pkg_gen_icmp_description = Erlang interface to ICMP sockets |
| pkg_gen_icmp_homepage = https://github.com/msantos/gen_icmp |
| pkg_gen_icmp_fetch = git |
| pkg_gen_icmp_repo = https://github.com/msantos/gen_icmp |
| pkg_gen_icmp_commit = master |
| |
| PACKAGES += gen_leader |
| pkg_gen_leader_name = gen_leader |
| pkg_gen_leader_description = leader election behavior |
| pkg_gen_leader_homepage = https://github.com/garret-smith/gen_leader_revival |
| pkg_gen_leader_fetch = git |
| pkg_gen_leader_repo = https://github.com/garret-smith/gen_leader_revival |
| pkg_gen_leader_commit = master |
| |
| PACKAGES += gen_nb_server |
| pkg_gen_nb_server_name = gen_nb_server |
| pkg_gen_nb_server_description = OTP behavior for writing non-blocking servers |
| pkg_gen_nb_server_homepage = https://github.com/kevsmith/gen_nb_server |
| pkg_gen_nb_server_fetch = git |
| pkg_gen_nb_server_repo = https://github.com/kevsmith/gen_nb_server |
| pkg_gen_nb_server_commit = master |
| |
| PACKAGES += gen_paxos |
| pkg_gen_paxos_name = gen_paxos |
| pkg_gen_paxos_description = An Erlang/OTP-style implementation of the PAXOS distributed consensus protocol |
| pkg_gen_paxos_homepage = https://github.com/gburd/gen_paxos |
| pkg_gen_paxos_fetch = git |
| pkg_gen_paxos_repo = https://github.com/gburd/gen_paxos |
| pkg_gen_paxos_commit = master |
| |
| PACKAGES += gen_rpc |
| pkg_gen_rpc_name = gen_rpc |
| pkg_gen_rpc_description = A scalable RPC library for Erlang-VM based languages |
| pkg_gen_rpc_homepage = https://github.com/priestjim/gen_rpc.git |
| pkg_gen_rpc_fetch = git |
| pkg_gen_rpc_repo = https://github.com/priestjim/gen_rpc.git |
| pkg_gen_rpc_commit = master |
| |
| PACKAGES += gen_smtp |
| pkg_gen_smtp_name = gen_smtp |
| pkg_gen_smtp_description = A generic Erlang SMTP server and client that can be extended via callback modules |
| pkg_gen_smtp_homepage = https://github.com/Vagabond/gen_smtp |
| pkg_gen_smtp_fetch = git |
| pkg_gen_smtp_repo = https://github.com/Vagabond/gen_smtp |
| pkg_gen_smtp_commit = master |
| |
| PACKAGES += gen_tracker |
| pkg_gen_tracker_name = gen_tracker |
| pkg_gen_tracker_description = supervisor with ets handling of children and their metadata |
| pkg_gen_tracker_homepage = https://github.com/erlyvideo/gen_tracker |
| pkg_gen_tracker_fetch = git |
| pkg_gen_tracker_repo = https://github.com/erlyvideo/gen_tracker |
| pkg_gen_tracker_commit = master |
| |
| PACKAGES += gen_unix |
| pkg_gen_unix_name = gen_unix |
| pkg_gen_unix_description = Erlang Unix socket interface |
| pkg_gen_unix_homepage = https://github.com/msantos/gen_unix |
| pkg_gen_unix_fetch = git |
| pkg_gen_unix_repo = https://github.com/msantos/gen_unix |
| pkg_gen_unix_commit = master |
| |
| PACKAGES += geode |
| pkg_geode_name = geode |
| pkg_geode_description = geohash/proximity lookup in pure, uncut erlang. |
| pkg_geode_homepage = https://github.com/bradfordw/geode |
| pkg_geode_fetch = git |
| pkg_geode_repo = https://github.com/bradfordw/geode |
| pkg_geode_commit = master |
| |
| PACKAGES += getopt |
| pkg_getopt_name = getopt |
| pkg_getopt_description = Module to parse command line arguments using the GNU getopt syntax |
| pkg_getopt_homepage = https://github.com/jcomellas/getopt |
| pkg_getopt_fetch = git |
| pkg_getopt_repo = https://github.com/jcomellas/getopt |
| pkg_getopt_commit = master |
| |
| PACKAGES += gettext |
| pkg_gettext_name = gettext |
| pkg_gettext_description = Erlang internationalization library. |
| pkg_gettext_homepage = https://github.com/etnt/gettext |
| pkg_gettext_fetch = git |
| pkg_gettext_repo = https://github.com/etnt/gettext |
| pkg_gettext_commit = master |
| |
| PACKAGES += giallo |
| pkg_giallo_name = giallo |
| pkg_giallo_description = Small and flexible web framework on top of Cowboy |
| pkg_giallo_homepage = https://github.com/kivra/giallo |
| pkg_giallo_fetch = git |
| pkg_giallo_repo = https://github.com/kivra/giallo |
| pkg_giallo_commit = master |
| |
| PACKAGES += gin |
| pkg_gin_name = gin |
| pkg_gin_description = The guards and for Erlang parse_transform |
| pkg_gin_homepage = https://github.com/mad-cocktail/gin |
| pkg_gin_fetch = git |
| pkg_gin_repo = https://github.com/mad-cocktail/gin |
| pkg_gin_commit = master |
| |
| PACKAGES += gitty |
| pkg_gitty_name = gitty |
| pkg_gitty_description = Git access in erlang |
| pkg_gitty_homepage = https://github.com/maxlapshin/gitty |
| pkg_gitty_fetch = git |
| pkg_gitty_repo = https://github.com/maxlapshin/gitty |
| pkg_gitty_commit = master |
| |
| PACKAGES += gpb |
| pkg_gpb_name = gpb |
| pkg_gpb_description = A Google Protobuf implementation for Erlang |
| pkg_gpb_homepage = https://github.com/tomas-abrahamsson/gpb |
| pkg_gpb_fetch = git |
| pkg_gpb_repo = https://github.com/tomas-abrahamsson/gpb |
| pkg_gpb_commit = master |
| |
| PACKAGES += gproc |
| pkg_gproc_name = gproc |
| pkg_gproc_description = Extended process registry for Erlang |
| pkg_gproc_homepage = https://github.com/uwiger/gproc |
| pkg_gproc_fetch = git |
| pkg_gproc_repo = https://github.com/uwiger/gproc |
| pkg_gproc_commit = master |
| |
| PACKAGES += grapherl |
| pkg_grapherl_name = grapherl |
| pkg_grapherl_description = Create graphs of Erlang systems and programs |
| pkg_grapherl_homepage = https://github.com/eproxus/grapherl |
| pkg_grapherl_fetch = git |
| pkg_grapherl_repo = https://github.com/eproxus/grapherl |
| pkg_grapherl_commit = master |
| |
| PACKAGES += grpc |
| pkg_grpc_name = grpc |
| pkg_grpc_description = gRPC server in Erlang |
| pkg_grpc_homepage = https://github.com/Bluehouse-Technology/grpc |
| pkg_grpc_fetch = git |
| pkg_grpc_repo = https://github.com/Bluehouse-Technology/grpc |
| pkg_grpc_commit = master |
| |
| PACKAGES += grpc_client |
| pkg_grpc_client_name = grpc_client |
| pkg_grpc_client_description = gRPC client in Erlang |
| pkg_grpc_client_homepage = https://github.com/Bluehouse-Technology/grpc_client |
| pkg_grpc_client_fetch = git |
| pkg_grpc_client_repo = https://github.com/Bluehouse-Technology/grpc_client |
| pkg_grpc_client_commit = master |
| |
| PACKAGES += gun |
| pkg_gun_name = gun |
| pkg_gun_description = Asynchronous SPDY, HTTP and Websocket client written in Erlang. |
| pkg_gun_homepage = http//ninenines.eu |
| pkg_gun_fetch = git |
| pkg_gun_repo = https://github.com/ninenines/gun |
| pkg_gun_commit = master |
| |
| PACKAGES += hackney |
| pkg_hackney_name = hackney |
| pkg_hackney_description = simple HTTP client in Erlang |
| pkg_hackney_homepage = https://github.com/benoitc/hackney |
| pkg_hackney_fetch = git |
| pkg_hackney_repo = https://github.com/benoitc/hackney |
| pkg_hackney_commit = master |
| |
| PACKAGES += hamcrest |
| pkg_hamcrest_name = hamcrest |
| pkg_hamcrest_description = Erlang port of Hamcrest |
| pkg_hamcrest_homepage = https://github.com/hyperthunk/hamcrest-erlang |
| pkg_hamcrest_fetch = git |
| pkg_hamcrest_repo = https://github.com/hyperthunk/hamcrest-erlang |
| pkg_hamcrest_commit = master |
| |
| PACKAGES += hottub |
| pkg_hottub_name = hottub |
| pkg_hottub_description = Permanent Erlang Worker Pool |
| pkg_hottub_homepage = https://github.com/bfrog/hottub |
| pkg_hottub_fetch = git |
| pkg_hottub_repo = https://github.com/bfrog/hottub |
| pkg_hottub_commit = master |
| |
| PACKAGES += hpack |
| pkg_hpack_name = hpack |
| pkg_hpack_description = HPACK Implementation for Erlang |
| pkg_hpack_homepage = https://github.com/joedevivo/hpack |
| pkg_hpack_fetch = git |
| pkg_hpack_repo = https://github.com/joedevivo/hpack |
| pkg_hpack_commit = master |
| |
| PACKAGES += hyper |
| pkg_hyper_name = hyper |
| pkg_hyper_description = Erlang implementation of HyperLogLog |
| pkg_hyper_homepage = https://github.com/GameAnalytics/hyper |
| pkg_hyper_fetch = git |
| pkg_hyper_repo = https://github.com/GameAnalytics/hyper |
| pkg_hyper_commit = master |
| |
| PACKAGES += i18n |
| pkg_i18n_name = i18n |
| pkg_i18n_description = International components for unicode from Erlang (unicode, date, string, number, format, locale, localization, transliteration, icu4e) |
| pkg_i18n_homepage = https://github.com/erlang-unicode/i18n |
| pkg_i18n_fetch = git |
| pkg_i18n_repo = https://github.com/erlang-unicode/i18n |
| pkg_i18n_commit = master |
| |
| PACKAGES += ibrowse |
| pkg_ibrowse_name = ibrowse |
| pkg_ibrowse_description = Erlang HTTP client |
| pkg_ibrowse_homepage = https://github.com/cmullaparthi/ibrowse |
| pkg_ibrowse_fetch = git |
| pkg_ibrowse_repo = https://github.com/cmullaparthi/ibrowse |
| pkg_ibrowse_commit = master |
| |
| PACKAGES += idna |
| pkg_idna_name = idna |
| pkg_idna_description = Erlang IDNA lib |
| pkg_idna_homepage = https://github.com/benoitc/erlang-idna |
| pkg_idna_fetch = git |
| pkg_idna_repo = https://github.com/benoitc/erlang-idna |
| pkg_idna_commit = master |
| |
| PACKAGES += irc_lib |
| pkg_irc_lib_name = irc_lib |
| pkg_irc_lib_description = Erlang irc client library |
| pkg_irc_lib_homepage = https://github.com/OtpChatBot/irc_lib |
| pkg_irc_lib_fetch = git |
| pkg_irc_lib_repo = https://github.com/OtpChatBot/irc_lib |
| pkg_irc_lib_commit = master |
| |
| PACKAGES += ircd |
| pkg_ircd_name = ircd |
| pkg_ircd_description = A pluggable IRC daemon application/library for Erlang. |
| pkg_ircd_homepage = https://github.com/tonyg/erlang-ircd |
| pkg_ircd_fetch = git |
| pkg_ircd_repo = https://github.com/tonyg/erlang-ircd |
| pkg_ircd_commit = master |
| |
| PACKAGES += iris |
| pkg_iris_name = iris |
| pkg_iris_description = Iris Erlang binding |
| pkg_iris_homepage = https://github.com/project-iris/iris-erl |
| pkg_iris_fetch = git |
| pkg_iris_repo = https://github.com/project-iris/iris-erl |
| pkg_iris_commit = master |
| |
| PACKAGES += iso8601 |
| pkg_iso8601_name = iso8601 |
| pkg_iso8601_description = Erlang ISO 8601 date formatter/parser |
| pkg_iso8601_homepage = https://github.com/seansawyer/erlang_iso8601 |
| pkg_iso8601_fetch = git |
| pkg_iso8601_repo = https://github.com/seansawyer/erlang_iso8601 |
| pkg_iso8601_commit = master |
| |
| PACKAGES += jamdb_sybase |
| pkg_jamdb_sybase_name = jamdb_sybase |
| pkg_jamdb_sybase_description = Erlang driver for SAP Sybase ASE |
| pkg_jamdb_sybase_homepage = https://github.com/erlangbureau/jamdb_sybase |
| pkg_jamdb_sybase_fetch = git |
| pkg_jamdb_sybase_repo = https://github.com/erlangbureau/jamdb_sybase |
| pkg_jamdb_sybase_commit = master |
| |
| PACKAGES += jesse |
| pkg_jesse_name = jesse |
| pkg_jesse_description = jesse (JSon Schema Erlang) is an implementation of a json schema validator for Erlang. |
| pkg_jesse_homepage = https://github.com/for-GET/jesse |
| pkg_jesse_fetch = git |
| pkg_jesse_repo = https://github.com/for-GET/jesse |
| pkg_jesse_commit = master |
| |
| PACKAGES += jiffy |
| pkg_jiffy_name = jiffy |
| pkg_jiffy_description = JSON NIFs for Erlang. |
| pkg_jiffy_homepage = https://github.com/davisp/jiffy |
| pkg_jiffy_fetch = git |
| pkg_jiffy_repo = https://github.com/davisp/jiffy |
| pkg_jiffy_commit = master |
| |
| PACKAGES += jiffy_v |
| pkg_jiffy_v_name = jiffy_v |
| pkg_jiffy_v_description = JSON validation utility |
| pkg_jiffy_v_homepage = https://github.com/shizzard/jiffy-v |
| pkg_jiffy_v_fetch = git |
| pkg_jiffy_v_repo = https://github.com/shizzard/jiffy-v |
| pkg_jiffy_v_commit = master |
| |
| PACKAGES += jobs |
| pkg_jobs_name = jobs |
| pkg_jobs_description = Job scheduler for load regulation |
| pkg_jobs_homepage = https://github.com/uwiger/jobs |
| pkg_jobs_fetch = git |
| pkg_jobs_repo = https://github.com/uwiger/jobs |
| pkg_jobs_commit = master |
| |
| PACKAGES += joxa |
| pkg_joxa_name = joxa |
| pkg_joxa_description = A Modern Lisp for the Erlang VM |
| pkg_joxa_homepage = https://github.com/joxa/joxa |
| pkg_joxa_fetch = git |
| pkg_joxa_repo = https://github.com/joxa/joxa |
| pkg_joxa_commit = master |
| |
| PACKAGES += json_rec |
| pkg_json_rec_name = json_rec |
| pkg_json_rec_description = JSON to erlang record |
| pkg_json_rec_homepage = https://github.com/justinkirby/json_rec |
| pkg_json_rec_fetch = git |
| pkg_json_rec_repo = https://github.com/justinkirby/json_rec |
| pkg_json_rec_commit = master |
| |
| PACKAGES += jsone |
| pkg_jsone_name = jsone |
| pkg_jsone_description = An Erlang library for encoding, decoding JSON data. |
| pkg_jsone_homepage = https://github.com/sile/jsone.git |
| pkg_jsone_fetch = git |
| pkg_jsone_repo = https://github.com/sile/jsone.git |
| pkg_jsone_commit = master |
| |
| PACKAGES += jsonpath |
| pkg_jsonpath_name = jsonpath |
| pkg_jsonpath_description = Fast Erlang JSON data retrieval and updates via javascript-like notation |
| pkg_jsonpath_homepage = https://github.com/GeneStevens/jsonpath |
| pkg_jsonpath_fetch = git |
| pkg_jsonpath_repo = https://github.com/GeneStevens/jsonpath |
| pkg_jsonpath_commit = master |
| |
| PACKAGES += jsonx |
| pkg_jsonx_name = jsonx |
| pkg_jsonx_description = JSONX is an Erlang library for efficient decode and encode JSON, written in C. |
| pkg_jsonx_homepage = https://github.com/iskra/jsonx |
| pkg_jsonx_fetch = git |
| pkg_jsonx_repo = https://github.com/iskra/jsonx |
| pkg_jsonx_commit = master |
| |
| PACKAGES += jsx |
| pkg_jsx_name = jsx |
| pkg_jsx_description = An Erlang application for consuming, producing and manipulating JSON. |
| pkg_jsx_homepage = https://github.com/talentdeficit/jsx |
| pkg_jsx_fetch = git |
| pkg_jsx_repo = https://github.com/talentdeficit/jsx |
| pkg_jsx_commit = main |
| |
| PACKAGES += kafka_protocol |
| pkg_kafka_protocol_name = kafka_protocol |
| pkg_kafka_protocol_description = Kafka protocol Erlang library |
| pkg_kafka_protocol_homepage = https://github.com/kafka4beam/kafka_protocol |
| pkg_kafka_protocol_fetch = git |
| pkg_kafka_protocol_repo = https://github.com/kafka4beam/kafka_protocol |
| pkg_kafka_protocol_commit = master |
| |
| PACKAGES += kai |
| pkg_kai_name = kai |
| pkg_kai_description = DHT storage by Takeshi Inoue |
| pkg_kai_homepage = https://github.com/synrc/kai |
| pkg_kai_fetch = git |
| pkg_kai_repo = https://github.com/synrc/kai |
| pkg_kai_commit = master |
| |
| PACKAGES += katja |
| pkg_katja_name = katja |
| pkg_katja_description = A simple Riemann client written in Erlang. |
| pkg_katja_homepage = https://github.com/nifoc/katja |
| pkg_katja_fetch = git |
| pkg_katja_repo = https://github.com/nifoc/katja |
| pkg_katja_commit = master |
| |
| PACKAGES += key2value |
| pkg_key2value_name = key2value |
| pkg_key2value_description = Erlang 2-way map |
| pkg_key2value_homepage = https://github.com/okeuday/key2value |
| pkg_key2value_fetch = git |
| pkg_key2value_repo = https://github.com/okeuday/key2value |
| pkg_key2value_commit = master |
| |
| PACKAGES += keys1value |
| pkg_keys1value_name = keys1value |
| pkg_keys1value_description = Erlang set associative map for key lists |
| pkg_keys1value_homepage = https://github.com/okeuday/keys1value |
| pkg_keys1value_fetch = git |
| pkg_keys1value_repo = https://github.com/okeuday/keys1value |
| pkg_keys1value_commit = master |
| |
| PACKAGES += kinetic |
| pkg_kinetic_name = kinetic |
| pkg_kinetic_description = Erlang Kinesis Client |
| pkg_kinetic_homepage = https://github.com/AdRoll/kinetic |
| pkg_kinetic_fetch = git |
| pkg_kinetic_repo = https://github.com/AdRoll/kinetic |
| pkg_kinetic_commit = main |
| |
| PACKAGES += kjell |
| pkg_kjell_name = kjell |
| pkg_kjell_description = Erlang Shell |
| pkg_kjell_homepage = https://github.com/karlll/kjell |
| pkg_kjell_fetch = git |
| pkg_kjell_repo = https://github.com/karlll/kjell |
| pkg_kjell_commit = master |
| |
| PACKAGES += kraken |
| pkg_kraken_name = kraken |
| pkg_kraken_description = Distributed Pubsub Server for Realtime Apps |
| pkg_kraken_homepage = https://github.com/Asana/kraken |
| pkg_kraken_fetch = git |
| pkg_kraken_repo = https://github.com/Asana/kraken |
| pkg_kraken_commit = master |
| |
| PACKAGES += kucumberl |
| pkg_kucumberl_name = kucumberl |
| pkg_kucumberl_description = A pure-erlang, open-source, implementation of Cucumber |
| pkg_kucumberl_homepage = https://github.com/openshine/kucumberl |
| pkg_kucumberl_fetch = git |
| pkg_kucumberl_repo = https://github.com/openshine/kucumberl |
| pkg_kucumberl_commit = master |
| |
| PACKAGES += kvc |
| pkg_kvc_name = kvc |
| pkg_kvc_description = KVC - Key Value Coding for Erlang data structures |
| pkg_kvc_homepage = https://github.com/etrepum/kvc |
| pkg_kvc_fetch = git |
| pkg_kvc_repo = https://github.com/etrepum/kvc |
| pkg_kvc_commit = master |
| |
| PACKAGES += kvlists |
| pkg_kvlists_name = kvlists |
| pkg_kvlists_description = Lists of key-value pairs (decoded JSON) in Erlang |
| pkg_kvlists_homepage = https://github.com/jcomellas/kvlists |
| pkg_kvlists_fetch = git |
| pkg_kvlists_repo = https://github.com/jcomellas/kvlists |
| pkg_kvlists_commit = master |
| |
| PACKAGES += kvs |
| pkg_kvs_name = kvs |
| pkg_kvs_description = Container and Iterator |
| pkg_kvs_homepage = https://github.com/synrc/kvs |
| pkg_kvs_fetch = git |
| pkg_kvs_repo = https://github.com/synrc/kvs |
| pkg_kvs_commit = master |
| |
| PACKAGES += lager |
| pkg_lager_name = lager |
| pkg_lager_description = A logging framework for Erlang/OTP. |
| pkg_lager_homepage = https://github.com/erlang-lager/lager |
| pkg_lager_fetch = git |
| pkg_lager_repo = https://github.com/erlang-lager/lager |
| pkg_lager_commit = master |
| |
| PACKAGES += lager_syslog |
| pkg_lager_syslog_name = lager_syslog |
| pkg_lager_syslog_description = Syslog backend for lager |
| pkg_lager_syslog_homepage = https://github.com/erlang-lager/lager_syslog |
| pkg_lager_syslog_fetch = git |
| pkg_lager_syslog_repo = https://github.com/erlang-lager/lager_syslog |
| pkg_lager_syslog_commit = master |
| |
| PACKAGES += lasse |
| pkg_lasse_name = lasse |
| pkg_lasse_description = SSE handler for Cowboy |
| pkg_lasse_homepage = https://github.com/inaka/lasse |
| pkg_lasse_fetch = git |
| pkg_lasse_repo = https://github.com/inaka/lasse |
| pkg_lasse_commit = master |
| |
| PACKAGES += ldap |
| pkg_ldap_name = ldap |
| pkg_ldap_description = LDAP server written in Erlang |
| pkg_ldap_homepage = https://github.com/spawnproc/ldap |
| pkg_ldap_fetch = git |
| pkg_ldap_repo = https://github.com/spawnproc/ldap |
| pkg_ldap_commit = master |
| |
| PACKAGES += lfe |
| pkg_lfe_name = lfe |
| pkg_lfe_description = Lisp Flavoured Erlang (LFE) |
| pkg_lfe_homepage = https://github.com/rvirding/lfe |
| pkg_lfe_fetch = git |
| pkg_lfe_repo = https://github.com/rvirding/lfe |
| pkg_lfe_commit = master |
| |
| PACKAGES += live |
| pkg_live_name = live |
| pkg_live_description = Automated module and configuration reloader. |
| pkg_live_homepage = http://ninenines.eu |
| pkg_live_fetch = git |
| pkg_live_repo = https://github.com/ninenines/live |
| pkg_live_commit = master |
| |
| PACKAGES += locker |
| pkg_locker_name = locker |
| pkg_locker_description = Atomic distributed 'check and set' for short-lived keys |
| pkg_locker_homepage = https://github.com/wooga/locker |
| pkg_locker_fetch = git |
| pkg_locker_repo = https://github.com/wooga/locker |
| pkg_locker_commit = master |
| |
| PACKAGES += locks |
| pkg_locks_name = locks |
| pkg_locks_description = A scalable, deadlock-resolving resource locker |
| pkg_locks_homepage = https://github.com/uwiger/locks |
| pkg_locks_fetch = git |
| pkg_locks_repo = https://github.com/uwiger/locks |
| pkg_locks_commit = master |
| |
| PACKAGES += log4erl |
| pkg_log4erl_name = log4erl |
| pkg_log4erl_description = A logger for erlang in the spirit of Log4J. |
| pkg_log4erl_homepage = https://github.com/ahmednawras/log4erl |
| pkg_log4erl_fetch = git |
| pkg_log4erl_repo = https://github.com/ahmednawras/log4erl |
| pkg_log4erl_commit = master |
| |
| PACKAGES += lol |
| pkg_lol_name = lol |
| pkg_lol_description = Lisp on erLang, and programming is fun again |
| pkg_lol_homepage = https://github.com/b0oh/lol |
| pkg_lol_fetch = git |
| pkg_lol_repo = https://github.com/b0oh/lol |
| pkg_lol_commit = master |
| |
| PACKAGES += lucid |
| pkg_lucid_name = lucid |
| pkg_lucid_description = HTTP/2 server written in Erlang |
| pkg_lucid_homepage = https://github.com/tatsuhiro-t/lucid |
| pkg_lucid_fetch = git |
| pkg_lucid_repo = https://github.com/tatsuhiro-t/lucid |
| pkg_lucid_commit = master |
| |
| PACKAGES += luerl |
| pkg_luerl_name = luerl |
| pkg_luerl_description = Lua in Erlang |
| pkg_luerl_homepage = https://github.com/rvirding/luerl |
| pkg_luerl_fetch = git |
| pkg_luerl_repo = https://github.com/rvirding/luerl |
| pkg_luerl_commit = develop |
| |
| PACKAGES += lux |
| pkg_lux_name = lux |
| pkg_lux_description = Lux (LUcid eXpect scripting) simplifies test automation and provides an Expect-style execution of commands |
| pkg_lux_homepage = https://github.com/hawk/lux |
| pkg_lux_fetch = git |
| pkg_lux_repo = https://github.com/hawk/lux |
| pkg_lux_commit = master |
| |
| PACKAGES += mad |
| pkg_mad_name = mad |
| pkg_mad_description = Small and Fast Rebar Replacement |
| pkg_mad_homepage = https://github.com/synrc/mad |
| pkg_mad_fetch = git |
| pkg_mad_repo = https://github.com/synrc/mad |
| pkg_mad_commit = master |
| |
| PACKAGES += marina |
| pkg_marina_name = marina |
| pkg_marina_description = Non-blocking Erlang Cassandra CQL3 client |
| pkg_marina_homepage = https://github.com/lpgauth/marina |
| pkg_marina_fetch = git |
| pkg_marina_repo = https://github.com/lpgauth/marina |
| pkg_marina_commit = master |
| |
| PACKAGES += mavg |
| pkg_mavg_name = mavg |
| pkg_mavg_description = Erlang :: Exponential moving average library |
| pkg_mavg_homepage = https://github.com/EchoTeam/mavg |
| pkg_mavg_fetch = git |
| pkg_mavg_repo = https://github.com/EchoTeam/mavg |
| pkg_mavg_commit = master |
| |
| PACKAGES += meck |
| pkg_meck_name = meck |
| pkg_meck_description = A mocking library for Erlang |
| pkg_meck_homepage = https://github.com/eproxus/meck |
| pkg_meck_fetch = git |
| pkg_meck_repo = https://github.com/eproxus/meck |
| pkg_meck_commit = master |
| |
| PACKAGES += mekao |
| pkg_mekao_name = mekao |
| pkg_mekao_description = SQL constructor |
| pkg_mekao_homepage = https://github.com/ddosia/mekao |
| pkg_mekao_fetch = git |
| pkg_mekao_repo = https://github.com/ddosia/mekao |
| pkg_mekao_commit = master |
| |
| PACKAGES += merl |
| pkg_merl_name = merl |
| pkg_merl_description = Metaprogramming in Erlang |
| pkg_merl_homepage = https://github.com/richcarl/merl |
| pkg_merl_fetch = git |
| pkg_merl_repo = https://github.com/richcarl/merl |
| pkg_merl_commit = master |
| |
| PACKAGES += mimerl |
| pkg_mimerl_name = mimerl |
| pkg_mimerl_description = library to handle mimetypes |
| pkg_mimerl_homepage = https://github.com/benoitc/mimerl |
| pkg_mimerl_fetch = git |
| pkg_mimerl_repo = https://github.com/benoitc/mimerl |
| pkg_mimerl_commit = master |
| |
| PACKAGES += mimetypes |
| pkg_mimetypes_name = mimetypes |
| pkg_mimetypes_description = Erlang MIME types library |
| pkg_mimetypes_homepage = https://github.com/spawngrid/mimetypes |
| pkg_mimetypes_fetch = git |
| pkg_mimetypes_repo = https://github.com/spawngrid/mimetypes |
| pkg_mimetypes_commit = master |
| |
| PACKAGES += mixer |
| pkg_mixer_name = mixer |
| pkg_mixer_description = Mix in functions from other modules |
| pkg_mixer_homepage = https://github.com/chef/mixer |
| pkg_mixer_fetch = git |
| pkg_mixer_repo = https://github.com/chef/mixer |
| pkg_mixer_commit = main |
| |
| PACKAGES += mochiweb |
| pkg_mochiweb_name = mochiweb |
| pkg_mochiweb_description = MochiWeb is an Erlang library for building lightweight HTTP servers. |
| pkg_mochiweb_homepage = https://github.com/mochi/mochiweb |
| pkg_mochiweb_fetch = git |
| pkg_mochiweb_repo = https://github.com/mochi/mochiweb |
| pkg_mochiweb_commit = main |
| |
| PACKAGES += mochiweb_xpath |
| pkg_mochiweb_xpath_name = mochiweb_xpath |
| pkg_mochiweb_xpath_description = XPath support for mochiweb's html parser |
| pkg_mochiweb_xpath_homepage = https://github.com/retnuh/mochiweb_xpath |
| pkg_mochiweb_xpath_fetch = git |
| pkg_mochiweb_xpath_repo = https://github.com/retnuh/mochiweb_xpath |
| pkg_mochiweb_xpath_commit = master |
| |
| PACKAGES += mockgyver |
| pkg_mockgyver_name = mockgyver |
| pkg_mockgyver_description = A mocking library for Erlang |
| pkg_mockgyver_homepage = https://github.com/klajo/mockgyver |
| pkg_mockgyver_fetch = git |
| pkg_mockgyver_repo = https://github.com/klajo/mockgyver |
| pkg_mockgyver_commit = master |
| |
| PACKAGES += modlib |
| pkg_modlib_name = modlib |
| pkg_modlib_description = Web framework based on Erlang's inets httpd |
| pkg_modlib_homepage = https://github.com/gar1t/modlib |
| pkg_modlib_fetch = git |
| pkg_modlib_repo = https://github.com/gar1t/modlib |
| pkg_modlib_commit = master |
| |
| PACKAGES += mongodb |
| pkg_mongodb_name = mongodb |
| pkg_mongodb_description = MongoDB driver for Erlang |
| pkg_mongodb_homepage = https://github.com/comtihon/mongodb-erlang |
| pkg_mongodb_fetch = git |
| pkg_mongodb_repo = https://github.com/comtihon/mongodb-erlang |
| pkg_mongodb_commit = master |
| |
| PACKAGES += mongooseim |
| pkg_mongooseim_name = mongooseim |
| pkg_mongooseim_description = Jabber / XMPP server with focus on performance and scalability, by Erlang Solutions |
| pkg_mongooseim_homepage = https://www.erlang-solutions.com/products/mongooseim-massively-scalable-ejabberd-platform |
| pkg_mongooseim_fetch = git |
| pkg_mongooseim_repo = https://github.com/esl/MongooseIM |
| pkg_mongooseim_commit = master |
| |
| PACKAGES += moyo |
| pkg_moyo_name = moyo |
| pkg_moyo_description = Erlang utility functions library |
| pkg_moyo_homepage = https://github.com/dwango/moyo |
| pkg_moyo_fetch = git |
| pkg_moyo_repo = https://github.com/dwango/moyo |
| pkg_moyo_commit = master |
| |
| PACKAGES += msgpack |
| pkg_msgpack_name = msgpack |
| pkg_msgpack_description = MessagePack (de)serializer implementation for Erlang |
| pkg_msgpack_homepage = https://github.com/msgpack/msgpack-erlang |
| pkg_msgpack_fetch = git |
| pkg_msgpack_repo = https://github.com/msgpack/msgpack-erlang |
| pkg_msgpack_commit = master |
| |
| PACKAGES += mu2 |
| pkg_mu2_name = mu2 |
| pkg_mu2_description = Erlang mutation testing tool |
| pkg_mu2_homepage = https://github.com/ramsay-t/mu2 |
| pkg_mu2_fetch = git |
| pkg_mu2_repo = https://github.com/ramsay-t/mu2 |
| pkg_mu2_commit = master |
| |
| PACKAGES += mustache |
| pkg_mustache_name = mustache |
| pkg_mustache_description = Mustache template engine for Erlang. |
| pkg_mustache_homepage = https://github.com/mojombo/mustache.erl |
| pkg_mustache_fetch = git |
| pkg_mustache_repo = https://github.com/mojombo/mustache.erl |
| pkg_mustache_commit = master |
| |
| PACKAGES += myproto |
| pkg_myproto_name = myproto |
| pkg_myproto_description = MySQL Server Protocol in Erlang |
| pkg_myproto_homepage = https://github.com/altenwald/myproto |
| pkg_myproto_fetch = git |
| pkg_myproto_repo = https://github.com/altenwald/myproto |
| pkg_myproto_commit = master |
| |
| PACKAGES += mysql |
| pkg_mysql_name = mysql |
| pkg_mysql_description = MySQL client library for Erlang/OTP |
| pkg_mysql_homepage = https://github.com/mysql-otp/mysql-otp |
| pkg_mysql_fetch = git |
| pkg_mysql_repo = https://github.com/mysql-otp/mysql-otp |
| pkg_mysql_commit = 1.7.0 |
| |
| PACKAGES += n2o |
| pkg_n2o_name = n2o |
| pkg_n2o_description = WebSocket Application Server |
| pkg_n2o_homepage = https://github.com/5HT/n2o |
| pkg_n2o_fetch = git |
| pkg_n2o_repo = https://github.com/5HT/n2o |
| pkg_n2o_commit = master |
| |
| PACKAGES += nat_upnp |
| pkg_nat_upnp_name = nat_upnp |
| pkg_nat_upnp_description = Erlang library to map your internal port to an external using UNP IGD |
| pkg_nat_upnp_homepage = https://github.com/benoitc/nat_upnp |
| pkg_nat_upnp_fetch = git |
| pkg_nat_upnp_repo = https://github.com/benoitc/nat_upnp |
| pkg_nat_upnp_commit = master |
| |
| PACKAGES += neo4j |
| pkg_neo4j_name = neo4j |
| pkg_neo4j_description = Erlang client library for Neo4J. |
| pkg_neo4j_homepage = https://github.com/dmitriid/neo4j-erlang |
| pkg_neo4j_fetch = git |
| pkg_neo4j_repo = https://github.com/dmitriid/neo4j-erlang |
| pkg_neo4j_commit = master |
| |
| PACKAGES += neotoma |
| pkg_neotoma_name = neotoma |
| pkg_neotoma_description = Erlang library and packrat parser-generator for parsing expression grammars. |
| pkg_neotoma_homepage = https://github.com/seancribbs/neotoma |
| pkg_neotoma_fetch = git |
| pkg_neotoma_repo = https://github.com/seancribbs/neotoma |
| pkg_neotoma_commit = master |
| |
| PACKAGES += nifty |
| pkg_nifty_name = nifty |
| pkg_nifty_description = Erlang NIF wrapper generator |
| pkg_nifty_homepage = https://github.com/parapluu/nifty |
| pkg_nifty_fetch = git |
| pkg_nifty_repo = https://github.com/parapluu/nifty |
| pkg_nifty_commit = master |
| |
| PACKAGES += nitrogen_core |
| pkg_nitrogen_core_name = nitrogen_core |
| pkg_nitrogen_core_description = The core Nitrogen library. |
| pkg_nitrogen_core_homepage = http://nitrogenproject.com/ |
| pkg_nitrogen_core_fetch = git |
| pkg_nitrogen_core_repo = https://github.com/nitrogen/nitrogen_core |
| pkg_nitrogen_core_commit = master |
| |
| PACKAGES += nkpacket |
| pkg_nkpacket_name = nkpacket |
| pkg_nkpacket_description = Generic Erlang transport layer |
| pkg_nkpacket_homepage = https://github.com/Nekso/nkpacket |
| pkg_nkpacket_fetch = git |
| pkg_nkpacket_repo = https://github.com/Nekso/nkpacket |
| pkg_nkpacket_commit = master |
| |
| PACKAGES += nksip |
| pkg_nksip_name = nksip |
| pkg_nksip_description = Erlang SIP application server |
| pkg_nksip_homepage = https://github.com/kalta/nksip |
| pkg_nksip_fetch = git |
| pkg_nksip_repo = https://github.com/kalta/nksip |
| pkg_nksip_commit = master |
| |
| PACKAGES += nodefinder |
| pkg_nodefinder_name = nodefinder |
| pkg_nodefinder_description = automatic node discovery via UDP multicast |
| pkg_nodefinder_homepage = https://github.com/erlanger/nodefinder |
| pkg_nodefinder_fetch = git |
| pkg_nodefinder_repo = https://github.com/okeuday/nodefinder |
| pkg_nodefinder_commit = master |
| |
| PACKAGES += nprocreg |
| pkg_nprocreg_name = nprocreg |
| pkg_nprocreg_description = Minimal Distributed Erlang Process Registry |
| pkg_nprocreg_homepage = http://nitrogenproject.com/ |
| pkg_nprocreg_fetch = git |
| pkg_nprocreg_repo = https://github.com/nitrogen/nprocreg |
| pkg_nprocreg_commit = master |
| |
| PACKAGES += oauth |
| pkg_oauth_name = oauth |
| pkg_oauth_description = An Erlang OAuth 1.0 implementation |
| pkg_oauth_homepage = https://github.com/tim/erlang-oauth |
| pkg_oauth_fetch = git |
| pkg_oauth_repo = https://github.com/tim/erlang-oauth |
| pkg_oauth_commit = main |
| |
| PACKAGES += oauth2 |
| pkg_oauth2_name = oauth2 |
| pkg_oauth2_description = Erlang Oauth2 implementation |
| pkg_oauth2_homepage = https://github.com/kivra/oauth2 |
| pkg_oauth2_fetch = git |
| pkg_oauth2_repo = https://github.com/kivra/oauth2 |
| pkg_oauth2_commit = master |
| |
| PACKAGES += observer_cli |
| pkg_observer_cli_name = observer_cli |
| pkg_observer_cli_description = Visualize Erlang/Elixir Nodes On The Command Line |
| pkg_observer_cli_homepage = http://zhongwencool.github.io/observer_cli |
| pkg_observer_cli_fetch = git |
| pkg_observer_cli_repo = https://github.com/zhongwencool/observer_cli |
| pkg_observer_cli_commit = master |
| |
| PACKAGES += octopus |
| pkg_octopus_name = octopus |
| pkg_octopus_description = Small and flexible pool manager written in Erlang |
| pkg_octopus_homepage = https://github.com/erlangbureau/octopus |
| pkg_octopus_fetch = git |
| pkg_octopus_repo = https://github.com/erlangbureau/octopus |
| pkg_octopus_commit = master |
| |
| PACKAGES += openflow |
| pkg_openflow_name = openflow |
| pkg_openflow_description = An OpenFlow controller written in pure erlang |
| pkg_openflow_homepage = https://github.com/renatoaguiar/erlang-openflow |
| pkg_openflow_fetch = git |
| pkg_openflow_repo = https://github.com/renatoaguiar/erlang-openflow |
| pkg_openflow_commit = master |
| |
| PACKAGES += openid |
| pkg_openid_name = openid |
| pkg_openid_description = Erlang OpenID |
| pkg_openid_homepage = https://github.com/brendonh/erl_openid |
| pkg_openid_fetch = git |
| pkg_openid_repo = https://github.com/brendonh/erl_openid |
| pkg_openid_commit = master |
| |
| PACKAGES += openpoker |
| pkg_openpoker_name = openpoker |
| pkg_openpoker_description = Genesis Texas hold'em Game Server |
| pkg_openpoker_homepage = https://github.com/hpyhacking/openpoker |
| pkg_openpoker_fetch = git |
| pkg_openpoker_repo = https://github.com/hpyhacking/openpoker |
| pkg_openpoker_commit = master |
| |
| PACKAGES += otpbp |
| pkg_otpbp_name = otpbp |
| pkg_otpbp_description = Parse transformer for use new OTP functions in old Erlang/OTP releases (R15, R16, 17, 18, 19) |
| pkg_otpbp_homepage = https://github.com/Ledest/otpbp |
| pkg_otpbp_fetch = git |
| pkg_otpbp_repo = https://github.com/Ledest/otpbp |
| pkg_otpbp_commit = master |
| |
| PACKAGES += pal |
| pkg_pal_name = pal |
| pkg_pal_description = Pragmatic Authentication Library |
| pkg_pal_homepage = https://github.com/manifest/pal |
| pkg_pal_fetch = git |
| pkg_pal_repo = https://github.com/manifest/pal |
| pkg_pal_commit = master |
| |
| PACKAGES += parse_trans |
| pkg_parse_trans_name = parse_trans |
| pkg_parse_trans_description = Parse transform utilities for Erlang |
| pkg_parse_trans_homepage = https://github.com/uwiger/parse_trans |
| pkg_parse_trans_fetch = git |
| pkg_parse_trans_repo = https://github.com/uwiger/parse_trans |
| pkg_parse_trans_commit = master |
| |
| PACKAGES += parsexml |
| pkg_parsexml_name = parsexml |
| pkg_parsexml_description = Simple DOM XML parser with convenient and very simple API |
| pkg_parsexml_homepage = https://github.com/maxlapshin/parsexml |
| pkg_parsexml_fetch = git |
| pkg_parsexml_repo = https://github.com/maxlapshin/parsexml |
| pkg_parsexml_commit = master |
| |
| PACKAGES += partisan |
| pkg_partisan_name = partisan |
| pkg_partisan_description = High-performance, high-scalability distributed computing with Erlang and Elixir. |
| pkg_partisan_homepage = http://partisan.cloud |
| pkg_partisan_fetch = git |
| pkg_partisan_repo = https://github.com/lasp-lang/partisan |
| pkg_partisan_commit = master |
| |
| PACKAGES += pegjs |
| pkg_pegjs_name = pegjs |
| pkg_pegjs_description = An implementation of PEG.js grammar for Erlang. |
| pkg_pegjs_homepage = https://github.com/dmitriid/pegjs |
| pkg_pegjs_fetch = git |
| pkg_pegjs_repo = https://github.com/dmitriid/pegjs |
| pkg_pegjs_commit = master |
| |
| PACKAGES += percept2 |
| pkg_percept2_name = percept2 |
| pkg_percept2_description = Concurrent profiling tool for Erlang |
| pkg_percept2_homepage = https://github.com/huiqing/percept2 |
| pkg_percept2_fetch = git |
| pkg_percept2_repo = https://github.com/huiqing/percept2 |
| pkg_percept2_commit = master |
| |
| PACKAGES += pgo |
| pkg_pgo_name = pgo |
| pkg_pgo_description = Erlang Postgres client and connection pool |
| pkg_pgo_homepage = https://github.com/erleans/pgo.git |
| pkg_pgo_fetch = git |
| pkg_pgo_repo = https://github.com/erleans/pgo.git |
| pkg_pgo_commit = main |
| |
| PACKAGES += pgsql |
| pkg_pgsql_name = pgsql |
| pkg_pgsql_description = Erlang PostgreSQL driver |
| pkg_pgsql_homepage = https://github.com/semiocast/pgsql |
| pkg_pgsql_fetch = git |
| pkg_pgsql_repo = https://github.com/semiocast/pgsql |
| pkg_pgsql_commit = master |
| |
| PACKAGES += pkgx |
| pkg_pkgx_name = pkgx |
| pkg_pkgx_description = Build .deb packages from Erlang releases |
| pkg_pkgx_homepage = https://github.com/arjan/pkgx |
| pkg_pkgx_fetch = git |
| pkg_pkgx_repo = https://github.com/arjan/pkgx |
| pkg_pkgx_commit = master |
| |
| PACKAGES += pkt |
| pkg_pkt_name = pkt |
| pkg_pkt_description = Erlang network protocol library |
| pkg_pkt_homepage = https://github.com/msantos/pkt |
| pkg_pkt_fetch = git |
| pkg_pkt_repo = https://github.com/msantos/pkt |
| pkg_pkt_commit = master |
| |
| PACKAGES += plain_fsm |
| pkg_plain_fsm_name = plain_fsm |
| pkg_plain_fsm_description = A behaviour/support library for writing plain Erlang FSMs. |
| pkg_plain_fsm_homepage = https://github.com/uwiger/plain_fsm |
| pkg_plain_fsm_fetch = git |
| pkg_plain_fsm_repo = https://github.com/uwiger/plain_fsm |
| pkg_plain_fsm_commit = master |
| |
| PACKAGES += pmod_transform |
| pkg_pmod_transform_name = pmod_transform |
| pkg_pmod_transform_description = Parse transform for parameterized modules |
| pkg_pmod_transform_homepage = https://github.com/erlang/pmod_transform |
| pkg_pmod_transform_fetch = git |
| pkg_pmod_transform_repo = https://github.com/erlang/pmod_transform |
| pkg_pmod_transform_commit = master |
| |
| PACKAGES += pobox |
| pkg_pobox_name = pobox |
| pkg_pobox_description = External buffer processes to protect against mailbox overflow in Erlang |
| pkg_pobox_homepage = https://github.com/ferd/pobox |
| pkg_pobox_fetch = git |
| pkg_pobox_repo = https://github.com/ferd/pobox |
| pkg_pobox_commit = master |
| |
| PACKAGES += ponos |
| pkg_ponos_name = ponos |
| pkg_ponos_description = ponos is a simple yet powerful load generator written in erlang |
| pkg_ponos_homepage = https://github.com/klarna/ponos |
| pkg_ponos_fetch = git |
| pkg_ponos_repo = https://github.com/klarna/ponos |
| pkg_ponos_commit = master |
| |
| PACKAGES += poolboy |
| pkg_poolboy_name = poolboy |
| pkg_poolboy_description = A hunky Erlang worker pool factory |
| pkg_poolboy_homepage = https://github.com/devinus/poolboy |
| pkg_poolboy_fetch = git |
| pkg_poolboy_repo = https://github.com/devinus/poolboy |
| pkg_poolboy_commit = master |
| |
| PACKAGES += pooler |
| pkg_pooler_name = pooler |
| pkg_pooler_description = An OTP Process Pool Application |
| pkg_pooler_homepage = https://github.com/seth/pooler |
| pkg_pooler_fetch = git |
| pkg_pooler_repo = https://github.com/seth/pooler |
| pkg_pooler_commit = master |
| |
| PACKAGES += pqueue |
| pkg_pqueue_name = pqueue |
| pkg_pqueue_description = Erlang Priority Queues |
| pkg_pqueue_homepage = https://github.com/okeuday/pqueue |
| pkg_pqueue_fetch = git |
| pkg_pqueue_repo = https://github.com/okeuday/pqueue |
| pkg_pqueue_commit = master |
| |
| PACKAGES += procket |
| pkg_procket_name = procket |
| pkg_procket_description = Erlang interface to low level socket operations |
| pkg_procket_homepage = http://blog.listincomprehension.com/search/label/procket |
| pkg_procket_fetch = git |
| pkg_procket_repo = https://github.com/msantos/procket |
| pkg_procket_commit = master |
| |
| PACKAGES += prometheus |
| pkg_prometheus_name = prometheus |
| pkg_prometheus_description = Prometheus.io client in Erlang |
| pkg_prometheus_homepage = https://github.com/deadtrickster/prometheus.erl |
| pkg_prometheus_fetch = git |
| pkg_prometheus_repo = https://github.com/deadtrickster/prometheus.erl |
| pkg_prometheus_commit = master |
| |
| PACKAGES += prop |
| pkg_prop_name = prop |
| pkg_prop_description = An Erlang code scaffolding and generator system. |
| pkg_prop_homepage = https://github.com/nuex/prop |
| pkg_prop_fetch = git |
| pkg_prop_repo = https://github.com/nuex/prop |
| pkg_prop_commit = master |
| |
| PACKAGES += proper |
| pkg_proper_name = proper |
| pkg_proper_description = PropEr: a QuickCheck-inspired property-based testing tool for Erlang. |
| pkg_proper_homepage = http://proper.softlab.ntua.gr |
| pkg_proper_fetch = git |
| pkg_proper_repo = https://github.com/manopapad/proper |
| pkg_proper_commit = master |
| |
| PACKAGES += props |
| pkg_props_name = props |
| pkg_props_description = Property structure library |
| pkg_props_homepage = https://github.com/greyarea/props |
| pkg_props_fetch = git |
| pkg_props_repo = https://github.com/greyarea/props |
| pkg_props_commit = master |
| |
| PACKAGES += protobuffs |
| pkg_protobuffs_name = protobuffs |
| pkg_protobuffs_description = An implementation of Google's Protocol Buffers for Erlang, based on ngerakines/erlang_protobuffs. |
| pkg_protobuffs_homepage = https://github.com/basho/erlang_protobuffs |
| pkg_protobuffs_fetch = git |
| pkg_protobuffs_repo = https://github.com/basho/erlang_protobuffs |
| pkg_protobuffs_commit = master |
| |
| PACKAGES += psycho |
| pkg_psycho_name = psycho |
| pkg_psycho_description = HTTP server that provides a WSGI-like interface for applications and middleware. |
| pkg_psycho_homepage = https://github.com/gar1t/psycho |
| pkg_psycho_fetch = git |
| pkg_psycho_repo = https://github.com/gar1t/psycho |
| pkg_psycho_commit = master |
| |
| PACKAGES += purity |
| pkg_purity_name = purity |
| pkg_purity_description = A side-effect analyzer for Erlang |
| pkg_purity_homepage = https://github.com/mpitid/purity |
| pkg_purity_fetch = git |
| pkg_purity_repo = https://github.com/mpitid/purity |
| pkg_purity_commit = master |
| |
| PACKAGES += qdate |
| pkg_qdate_name = qdate |
| pkg_qdate_description = Date, time, and timezone parsing, formatting, and conversion for Erlang. |
| pkg_qdate_homepage = https://github.com/choptastic/qdate |
| pkg_qdate_fetch = git |
| pkg_qdate_repo = https://github.com/choptastic/qdate |
| pkg_qdate_commit = master |
| |
| PACKAGES += qrcode |
| pkg_qrcode_name = qrcode |
| pkg_qrcode_description = QR Code encoder in Erlang |
| pkg_qrcode_homepage = https://github.com/komone/qrcode |
| pkg_qrcode_fetch = git |
| pkg_qrcode_repo = https://github.com/komone/qrcode |
| pkg_qrcode_commit = master |
| |
| PACKAGES += quest |
| pkg_quest_name = quest |
| pkg_quest_description = Learn Erlang through this set of challenges. An interactive system for getting to know Erlang. |
| pkg_quest_homepage = https://github.com/eriksoe/ErlangQuest |
| pkg_quest_fetch = git |
| pkg_quest_repo = https://github.com/eriksoe/ErlangQuest |
| pkg_quest_commit = master |
| |
| PACKAGES += quickrand |
| pkg_quickrand_name = quickrand |
| pkg_quickrand_description = Quick Erlang Random Number Generation |
| pkg_quickrand_homepage = https://github.com/okeuday/quickrand |
| pkg_quickrand_fetch = git |
| pkg_quickrand_repo = https://github.com/okeuday/quickrand |
| pkg_quickrand_commit = master |
| |
| PACKAGES += rabbit_exchange_type_riak |
| pkg_rabbit_exchange_type_riak_name = rabbit_exchange_type_riak |
| pkg_rabbit_exchange_type_riak_description = Custom RabbitMQ exchange type for sticking messages in Riak |
| pkg_rabbit_exchange_type_riak_homepage = https://github.com/jbrisbin/riak-exchange |
| pkg_rabbit_exchange_type_riak_fetch = git |
| pkg_rabbit_exchange_type_riak_repo = https://github.com/jbrisbin/riak-exchange |
| pkg_rabbit_exchange_type_riak_commit = master |
| |
| PACKAGES += rack |
| pkg_rack_name = rack |
| pkg_rack_description = Rack handler for erlang |
| pkg_rack_homepage = https://github.com/erlyvideo/rack |
| pkg_rack_fetch = git |
| pkg_rack_repo = https://github.com/erlyvideo/rack |
| pkg_rack_commit = master |
| |
| PACKAGES += radierl |
| pkg_radierl_name = radierl |
| pkg_radierl_description = RADIUS protocol stack implemented in Erlang. |
| pkg_radierl_homepage = https://github.com/vances/radierl |
| pkg_radierl_fetch = git |
| pkg_radierl_repo = https://github.com/vances/radierl |
| pkg_radierl_commit = master |
| |
| PACKAGES += ranch |
| pkg_ranch_name = ranch |
| pkg_ranch_description = Socket acceptor pool for TCP protocols. |
| pkg_ranch_homepage = http://ninenines.eu |
| pkg_ranch_fetch = git |
| pkg_ranch_repo = https://github.com/ninenines/ranch |
| pkg_ranch_commit = 1.2.1 |
| |
| PACKAGES += rbeacon |
| pkg_rbeacon_name = rbeacon |
| pkg_rbeacon_description = LAN discovery and presence in Erlang. |
| pkg_rbeacon_homepage = https://github.com/refuge/rbeacon |
| pkg_rbeacon_fetch = git |
| pkg_rbeacon_repo = https://github.com/refuge/rbeacon |
| pkg_rbeacon_commit = master |
| |
| PACKAGES += re2 |
| pkg_re2_name = re2 |
| pkg_re2_description = Erlang NIF bindings for RE2 regex library |
| pkg_re2_homepage = https://github.com/dukesoferl/re2 |
| pkg_re2_fetch = git |
| pkg_re2_repo = https://github.com/dukesoferl/re2 |
| pkg_re2_commit = master |
| |
| PACKAGES += rebus |
| pkg_rebus_name = rebus |
| pkg_rebus_description = A stupid simple, internal, pub/sub event bus written in- and for Erlang. |
| pkg_rebus_homepage = https://github.com/olle/rebus |
| pkg_rebus_fetch = git |
| pkg_rebus_repo = https://github.com/olle/rebus |
| pkg_rebus_commit = master |
| |
| PACKAGES += rec2json |
| pkg_rec2json_name = rec2json |
| pkg_rec2json_description = Compile erlang record definitions into modules to convert them to/from json easily. |
| pkg_rec2json_homepage = https://github.com/lordnull/rec2json |
| pkg_rec2json_fetch = git |
| pkg_rec2json_repo = https://github.com/lordnull/rec2json |
| pkg_rec2json_commit = master |
| |
| PACKAGES += recon |
| pkg_recon_name = recon |
| pkg_recon_description = Collection of functions and scripts to debug Erlang in production. |
| pkg_recon_homepage = https://github.com/ferd/recon |
| pkg_recon_fetch = git |
| pkg_recon_repo = https://github.com/ferd/recon |
| pkg_recon_commit = master |
| |
| PACKAGES += record_info |
| pkg_record_info_name = record_info |
| pkg_record_info_description = Convert between record and proplist |
| pkg_record_info_homepage = https://github.com/bipthelin/erlang-record_info |
| pkg_record_info_fetch = git |
| pkg_record_info_repo = https://github.com/bipthelin/erlang-record_info |
| pkg_record_info_commit = master |
| |
| PACKAGES += redgrid |
| pkg_redgrid_name = redgrid |
| pkg_redgrid_description = automatic Erlang node discovery via redis |
| pkg_redgrid_homepage = https://github.com/jkvor/redgrid |
| pkg_redgrid_fetch = git |
| pkg_redgrid_repo = https://github.com/jkvor/redgrid |
| pkg_redgrid_commit = master |
| |
| PACKAGES += redo |
| pkg_redo_name = redo |
| pkg_redo_description = pipelined erlang redis client |
| pkg_redo_homepage = https://github.com/jkvor/redo |
| pkg_redo_fetch = git |
| pkg_redo_repo = https://github.com/jkvor/redo |
| pkg_redo_commit = master |
| |
| PACKAGES += reload_mk |
| pkg_reload_mk_name = reload_mk |
| pkg_reload_mk_description = Live reload plugin for erlang.mk. |
| pkg_reload_mk_homepage = https://github.com/bullno1/reload.mk |
| pkg_reload_mk_fetch = git |
| pkg_reload_mk_repo = https://github.com/bullno1/reload.mk |
| pkg_reload_mk_commit = master |
| |
| PACKAGES += reltool_util |
| pkg_reltool_util_name = reltool_util |
| pkg_reltool_util_description = Erlang reltool utility functionality application |
| pkg_reltool_util_homepage = https://github.com/okeuday/reltool_util |
| pkg_reltool_util_fetch = git |
| pkg_reltool_util_repo = https://github.com/okeuday/reltool_util |
| pkg_reltool_util_commit = master |
| |
| PACKAGES += relx |
| pkg_relx_name = relx |
| pkg_relx_description = Sane, simple release creation for Erlang |
| pkg_relx_homepage = https://github.com/erlware/relx |
| pkg_relx_fetch = git |
| pkg_relx_repo = https://github.com/erlware/relx |
| pkg_relx_commit = main |
| |
| PACKAGES += resource_discovery |
| pkg_resource_discovery_name = resource_discovery |
| pkg_resource_discovery_description = An application used to dynamically discover resources present in an Erlang node cluster. |
| pkg_resource_discovery_homepage = http://erlware.org/ |
| pkg_resource_discovery_fetch = git |
| pkg_resource_discovery_repo = https://github.com/erlware/resource_discovery |
| pkg_resource_discovery_commit = master |
| |
| PACKAGES += restc |
| pkg_restc_name = restc |
| pkg_restc_description = Erlang Rest Client |
| pkg_restc_homepage = https://github.com/kivra/restclient |
| pkg_restc_fetch = git |
| pkg_restc_repo = https://github.com/kivra/restclient |
| pkg_restc_commit = master |
| |
| PACKAGES += rfc4627_jsonrpc |
| pkg_rfc4627_jsonrpc_name = rfc4627_jsonrpc |
| pkg_rfc4627_jsonrpc_description = Erlang RFC4627 (JSON) codec and JSON-RPC server implementation. |
| pkg_rfc4627_jsonrpc_homepage = https://github.com/tonyg/erlang-rfc4627 |
| pkg_rfc4627_jsonrpc_fetch = git |
| pkg_rfc4627_jsonrpc_repo = https://github.com/tonyg/erlang-rfc4627 |
| pkg_rfc4627_jsonrpc_commit = master |
| |
| PACKAGES += riak_core |
| pkg_riak_core_name = riak_core |
| pkg_riak_core_description = Distributed systems infrastructure used by Riak. |
| pkg_riak_core_homepage = https://github.com/basho/riak_core |
| pkg_riak_core_fetch = git |
| pkg_riak_core_repo = https://github.com/basho/riak_core |
| pkg_riak_core_commit = develop |
| |
| PACKAGES += riak_dt |
| pkg_riak_dt_name = riak_dt |
| pkg_riak_dt_description = Convergent replicated datatypes in Erlang |
| pkg_riak_dt_homepage = https://github.com/basho/riak_dt |
| pkg_riak_dt_fetch = git |
| pkg_riak_dt_repo = https://github.com/basho/riak_dt |
| pkg_riak_dt_commit = master |
| |
| PACKAGES += riak_ensemble |
| pkg_riak_ensemble_name = riak_ensemble |
| pkg_riak_ensemble_description = Multi-Paxos framework in Erlang |
| pkg_riak_ensemble_homepage = https://github.com/basho/riak_ensemble |
| pkg_riak_ensemble_fetch = git |
| pkg_riak_ensemble_repo = https://github.com/basho/riak_ensemble |
| pkg_riak_ensemble_commit = develop |
| |
| PACKAGES += riak_kv |
| pkg_riak_kv_name = riak_kv |
| pkg_riak_kv_description = Riak Key/Value Store |
| pkg_riak_kv_homepage = https://github.com/basho/riak_kv |
| pkg_riak_kv_fetch = git |
| pkg_riak_kv_repo = https://github.com/basho/riak_kv |
| pkg_riak_kv_commit = develop |
| |
| PACKAGES += riak_pipe |
| pkg_riak_pipe_name = riak_pipe |
| pkg_riak_pipe_description = Riak Pipelines |
| pkg_riak_pipe_homepage = https://github.com/basho/riak_pipe |
| pkg_riak_pipe_fetch = git |
| pkg_riak_pipe_repo = https://github.com/basho/riak_pipe |
| pkg_riak_pipe_commit = develop |
| |
| PACKAGES += riak_sysmon |
| pkg_riak_sysmon_name = riak_sysmon |
| pkg_riak_sysmon_description = Simple OTP app for managing Erlang VM system_monitor event messages |
| pkg_riak_sysmon_homepage = https://github.com/basho/riak_sysmon |
| pkg_riak_sysmon_fetch = git |
| pkg_riak_sysmon_repo = https://github.com/basho/riak_sysmon |
| pkg_riak_sysmon_commit = master |
| |
| PACKAGES += riakc |
| pkg_riakc_name = riakc |
| pkg_riakc_description = Erlang clients for Riak. |
| pkg_riakc_homepage = https://github.com/basho/riak-erlang-client |
| pkg_riakc_fetch = git |
| pkg_riakc_repo = https://github.com/basho/riak-erlang-client |
| pkg_riakc_commit = master |
| |
| PACKAGES += rlimit |
| pkg_rlimit_name = rlimit |
| pkg_rlimit_description = Magnus Klaar's rate limiter code from etorrent |
| pkg_rlimit_homepage = https://github.com/jlouis/rlimit |
| pkg_rlimit_fetch = git |
| pkg_rlimit_repo = https://github.com/jlouis/rlimit |
| pkg_rlimit_commit = master |
| |
| PACKAGES += rust_mk |
| pkg_rust_mk_name = rust_mk |
| pkg_rust_mk_description = Build Rust crates in an Erlang application |
| pkg_rust_mk_homepage = https://github.com/goertzenator/rust.mk |
| pkg_rust_mk_fetch = git |
| pkg_rust_mk_repo = https://github.com/goertzenator/rust.mk |
| pkg_rust_mk_commit = master |
| |
| PACKAGES += safetyvalve |
| pkg_safetyvalve_name = safetyvalve |
| pkg_safetyvalve_description = A safety valve for your erlang node |
| pkg_safetyvalve_homepage = https://github.com/jlouis/safetyvalve |
| pkg_safetyvalve_fetch = git |
| pkg_safetyvalve_repo = https://github.com/jlouis/safetyvalve |
| pkg_safetyvalve_commit = master |
| |
| PACKAGES += seestar |
| pkg_seestar_name = seestar |
| pkg_seestar_description = The Erlang client for Cassandra 1.2+ binary protocol |
| pkg_seestar_homepage = https://github.com/iamaleksey/seestar |
| pkg_seestar_fetch = git |
| pkg_seestar_repo = https://github.com/iamaleksey/seestar |
| pkg_seestar_commit = master |
| |
| PACKAGES += setup |
| pkg_setup_name = setup |
| pkg_setup_description = Generic setup utility for Erlang-based systems |
| pkg_setup_homepage = https://github.com/uwiger/setup |
| pkg_setup_fetch = git |
| pkg_setup_repo = https://github.com/uwiger/setup |
| pkg_setup_commit = master |
| |
| PACKAGES += sext |
| pkg_sext_name = sext |
| pkg_sext_description = Sortable Erlang Term Serialization |
| pkg_sext_homepage = https://github.com/uwiger/sext |
| pkg_sext_fetch = git |
| pkg_sext_repo = https://github.com/uwiger/sext |
| pkg_sext_commit = master |
| |
| PACKAGES += sfmt |
| pkg_sfmt_name = sfmt |
| pkg_sfmt_description = SFMT pseudo random number generator for Erlang. |
| pkg_sfmt_homepage = https://github.com/jj1bdx/sfmt-erlang |
| pkg_sfmt_fetch = git |
| pkg_sfmt_repo = https://github.com/jj1bdx/sfmt-erlang |
| pkg_sfmt_commit = master |
| |
| PACKAGES += sgte |
| pkg_sgte_name = sgte |
| pkg_sgte_description = A simple Erlang Template Engine |
| pkg_sgte_homepage = https://github.com/filippo/sgte |
| pkg_sgte_fetch = git |
| pkg_sgte_repo = https://github.com/filippo/sgte |
| pkg_sgte_commit = master |
| |
| PACKAGES += sheriff |
| pkg_sheriff_name = sheriff |
| pkg_sheriff_description = Parse transform for type based validation. |
| pkg_sheriff_homepage = http://ninenines.eu |
| pkg_sheriff_fetch = git |
| pkg_sheriff_repo = https://github.com/extend/sheriff |
| pkg_sheriff_commit = master |
| |
| PACKAGES += shotgun |
| pkg_shotgun_name = shotgun |
| pkg_shotgun_description = better than just a gun |
| pkg_shotgun_homepage = https://github.com/inaka/shotgun |
| pkg_shotgun_fetch = git |
| pkg_shotgun_repo = https://github.com/inaka/shotgun |
| pkg_shotgun_commit = master |
| |
| PACKAGES += sidejob |
| pkg_sidejob_name = sidejob |
| pkg_sidejob_description = Parallel worker and capacity limiting library for Erlang |
| pkg_sidejob_homepage = https://github.com/basho/sidejob |
| pkg_sidejob_fetch = git |
| pkg_sidejob_repo = https://github.com/basho/sidejob |
| pkg_sidejob_commit = develop |
| |
| PACKAGES += sieve |
| pkg_sieve_name = sieve |
| pkg_sieve_description = sieve is a simple TCP routing proxy (layer 7) in erlang |
| pkg_sieve_homepage = https://github.com/benoitc/sieve |
| pkg_sieve_fetch = git |
| pkg_sieve_repo = https://github.com/benoitc/sieve |
| pkg_sieve_commit = master |
| |
| PACKAGES += simhash |
| pkg_simhash_name = simhash |
| pkg_simhash_description = Simhashing for Erlang -- hashing algorithm to find near-duplicates in binary data. |
| pkg_simhash_homepage = https://github.com/ferd/simhash |
| pkg_simhash_fetch = git |
| pkg_simhash_repo = https://github.com/ferd/simhash |
| pkg_simhash_commit = master |
| |
| PACKAGES += simple_bridge |
| pkg_simple_bridge_name = simple_bridge |
| pkg_simple_bridge_description = A simple, standardized interface library to Erlang HTTP Servers. |
| pkg_simple_bridge_homepage = https://github.com/nitrogen/simple_bridge |
| pkg_simple_bridge_fetch = git |
| pkg_simple_bridge_repo = https://github.com/nitrogen/simple_bridge |
| pkg_simple_bridge_commit = master |
| |
| PACKAGES += simple_oauth2 |
| pkg_simple_oauth2_name = simple_oauth2 |
| pkg_simple_oauth2_description = Simple erlang OAuth2 client module for any http server framework (Google, Facebook, Yandex, Vkontakte are preconfigured) |
| pkg_simple_oauth2_homepage = https://github.com/virtan/simple_oauth2 |
| pkg_simple_oauth2_fetch = git |
| pkg_simple_oauth2_repo = https://github.com/virtan/simple_oauth2 |
| pkg_simple_oauth2_commit = master |
| |
| PACKAGES += skel |
| pkg_skel_name = skel |
| pkg_skel_description = A Streaming Process-based Skeleton Library for Erlang |
| pkg_skel_homepage = https://github.com/ParaPhrase/skel |
| pkg_skel_fetch = git |
| pkg_skel_repo = https://github.com/ParaPhrase/skel |
| pkg_skel_commit = master |
| |
| PACKAGES += slack |
| pkg_slack_name = slack |
| pkg_slack_description = Minimal slack notification OTP library. |
| pkg_slack_homepage = https://github.com/DonBranson/slack |
| pkg_slack_fetch = git |
| pkg_slack_repo = https://github.com/DonBranson/slack.git |
| pkg_slack_commit = master |
| |
| PACKAGES += snappyer |
| pkg_snappyer_name = snappyer |
| pkg_snappyer_description = Snappy as nif for Erlang |
| pkg_snappyer_homepage = https://github.com/zmstone/snappyer |
| pkg_snappyer_fetch = git |
| pkg_snappyer_repo = https://github.com/zmstone/snappyer.git |
| pkg_snappyer_commit = master |
| |
| PACKAGES += social |
| pkg_social_name = social |
| pkg_social_description = Cowboy handler for social login via OAuth2 providers |
| pkg_social_homepage = https://github.com/dvv/social |
| pkg_social_fetch = git |
| pkg_social_repo = https://github.com/dvv/social |
| pkg_social_commit = master |
| |
| PACKAGES += sqerl |
| pkg_sqerl_name = sqerl |
| pkg_sqerl_description = An Erlang-flavoured SQL DSL |
| pkg_sqerl_homepage = https://github.com/hairyhum/sqerl |
| pkg_sqerl_fetch = git |
| pkg_sqerl_repo = https://github.com/hairyhum/sqerl |
| pkg_sqerl_commit = master |
| |
| PACKAGES += srly |
| pkg_srly_name = srly |
| pkg_srly_description = Native Erlang Unix serial interface |
| pkg_srly_homepage = https://github.com/msantos/srly |
| pkg_srly_fetch = git |
| pkg_srly_repo = https://github.com/msantos/srly |
| pkg_srly_commit = master |
| |
| PACKAGES += sshrpc |
| pkg_sshrpc_name = sshrpc |
| pkg_sshrpc_description = Erlang SSH RPC module (experimental) |
| pkg_sshrpc_homepage = https://github.com/jj1bdx/sshrpc |
| pkg_sshrpc_fetch = git |
| pkg_sshrpc_repo = https://github.com/jj1bdx/sshrpc |
| pkg_sshrpc_commit = master |
| |
| PACKAGES += stable |
| pkg_stable_name = stable |
| pkg_stable_description = Library of assorted helpers for Cowboy web server. |
| pkg_stable_homepage = https://github.com/dvv/stable |
| pkg_stable_fetch = git |
| pkg_stable_repo = https://github.com/dvv/stable |
| pkg_stable_commit = master |
| |
| PACKAGES += statebox |
| pkg_statebox_name = statebox |
| pkg_statebox_description = Erlang state monad with merge/conflict-resolution capabilities. Useful for Riak. |
| pkg_statebox_homepage = https://github.com/mochi/statebox |
| pkg_statebox_fetch = git |
| pkg_statebox_repo = https://github.com/mochi/statebox |
| pkg_statebox_commit = master |
| |
| PACKAGES += statman |
| pkg_statman_name = statman |
| pkg_statman_description = Efficiently collect massive volumes of metrics inside the Erlang VM |
| pkg_statman_homepage = https://github.com/knutin/statman |
| pkg_statman_fetch = git |
| pkg_statman_repo = https://github.com/knutin/statman |
| pkg_statman_commit = master |
| |
| PACKAGES += statsderl |
| pkg_statsderl_name = statsderl |
| pkg_statsderl_description = StatsD client (erlang) |
| pkg_statsderl_homepage = https://github.com/lpgauth/statsderl |
| pkg_statsderl_fetch = git |
| pkg_statsderl_repo = https://github.com/lpgauth/statsderl |
| pkg_statsderl_commit = master |
| |
| PACKAGES += stdinout_pool |
| pkg_stdinout_pool_name = stdinout_pool |
| pkg_stdinout_pool_description = stdinout_pool : stuff goes in, stuff goes out. there's never any miscommunication. |
| pkg_stdinout_pool_homepage = https://github.com/mattsta/erlang-stdinout-pool |
| pkg_stdinout_pool_fetch = git |
| pkg_stdinout_pool_repo = https://github.com/mattsta/erlang-stdinout-pool |
| pkg_stdinout_pool_commit = master |
| |
| PACKAGES += stockdb |
| pkg_stockdb_name = stockdb |
| pkg_stockdb_description = Database for storing Stock Exchange quotes in erlang |
| pkg_stockdb_homepage = https://github.com/maxlapshin/stockdb |
| pkg_stockdb_fetch = git |
| pkg_stockdb_repo = https://github.com/maxlapshin/stockdb |
| pkg_stockdb_commit = master |
| |
| PACKAGES += subproc |
| pkg_subproc_name = subproc |
| pkg_subproc_description = unix subprocess manager with {active,once|false} modes |
| pkg_subproc_homepage = http://dozzie.jarowit.net/trac/wiki/subproc |
| pkg_subproc_fetch = git |
| pkg_subproc_repo = https://github.com/dozzie/subproc |
| pkg_subproc_commit = v0.1.0 |
| |
| PACKAGES += supervisor3 |
| pkg_supervisor3_name = supervisor3 |
| pkg_supervisor3_description = OTP supervisor with additional strategies |
| pkg_supervisor3_homepage = https://github.com/klarna/supervisor3 |
| pkg_supervisor3_fetch = git |
| pkg_supervisor3_repo = https://github.com/klarna/supervisor3.git |
| pkg_supervisor3_commit = master |
| |
| PACKAGES += swab |
| pkg_swab_name = swab |
| pkg_swab_description = General purpose buffer handling module |
| pkg_swab_homepage = https://github.com/crownedgrouse/swab |
| pkg_swab_fetch = git |
| pkg_swab_repo = https://github.com/crownedgrouse/swab |
| pkg_swab_commit = master |
| |
| PACKAGES += swarm |
| pkg_swarm_name = swarm |
| pkg_swarm_description = Fast and simple acceptor pool for Erlang |
| pkg_swarm_homepage = https://github.com/jeremey/swarm |
| pkg_swarm_fetch = git |
| pkg_swarm_repo = https://github.com/jeremey/swarm |
| pkg_swarm_commit = master |
| |
| PACKAGES += switchboard |
| pkg_switchboard_name = switchboard |
| pkg_switchboard_description = A framework for processing email using worker plugins. |
| pkg_switchboard_homepage = https://github.com/thusfresh/switchboard |
| pkg_switchboard_fetch = git |
| pkg_switchboard_repo = https://github.com/thusfresh/switchboard |
| pkg_switchboard_commit = master |
| |
| PACKAGES += syn |
| pkg_syn_name = syn |
| pkg_syn_description = A global Process Registry and Process Group manager for Erlang. |
| pkg_syn_homepage = https://github.com/ostinelli/syn |
| pkg_syn_fetch = git |
| pkg_syn_repo = https://github.com/ostinelli/syn |
| pkg_syn_commit = master |
| |
| PACKAGES += sync |
| pkg_sync_name = sync |
| pkg_sync_description = On-the-fly recompiling and reloading in Erlang. |
| pkg_sync_homepage = https://github.com/rustyio/sync |
| pkg_sync_fetch = git |
| pkg_sync_repo = https://github.com/rustyio/sync |
| pkg_sync_commit = master |
| |
| PACKAGES += syntaxerl |
| pkg_syntaxerl_name = syntaxerl |
| pkg_syntaxerl_description = Syntax checker for Erlang |
| pkg_syntaxerl_homepage = https://github.com/ten0s/syntaxerl |
| pkg_syntaxerl_fetch = git |
| pkg_syntaxerl_repo = https://github.com/ten0s/syntaxerl |
| pkg_syntaxerl_commit = master |
| |
| PACKAGES += syslog |
| pkg_syslog_name = syslog |
| pkg_syslog_description = Erlang port driver for interacting with syslog via syslog(3) |
| pkg_syslog_homepage = https://github.com/Vagabond/erlang-syslog |
| pkg_syslog_fetch = git |
| pkg_syslog_repo = https://github.com/Vagabond/erlang-syslog |
| pkg_syslog_commit = master |
| |
| PACKAGES += taskforce |
| pkg_taskforce_name = taskforce |
| pkg_taskforce_description = Erlang worker pools for controlled parallelisation of arbitrary tasks. |
| pkg_taskforce_homepage = https://github.com/g-andrade/taskforce |
| pkg_taskforce_fetch = git |
| pkg_taskforce_repo = https://github.com/g-andrade/taskforce |
| pkg_taskforce_commit = master |
| |
| PACKAGES += tddreloader |
| pkg_tddreloader_name = tddreloader |
| pkg_tddreloader_description = Shell utility for recompiling, reloading, and testing code as it changes |
| pkg_tddreloader_homepage = https://github.com/version2beta/tddreloader |
| pkg_tddreloader_fetch = git |
| pkg_tddreloader_repo = https://github.com/version2beta/tddreloader |
| pkg_tddreloader_commit = master |
| |
| PACKAGES += tempo |
| pkg_tempo_name = tempo |
| pkg_tempo_description = NIF-based date and time parsing and formatting for Erlang. |
| pkg_tempo_homepage = https://github.com/selectel/tempo |
| pkg_tempo_fetch = git |
| pkg_tempo_repo = https://github.com/selectel/tempo |
| pkg_tempo_commit = master |
| |
| PACKAGES += tinymq |
| pkg_tinymq_name = tinymq |
| pkg_tinymq_description = TinyMQ - a diminutive, in-memory message queue |
| pkg_tinymq_homepage = https://github.com/ChicagoBoss/tinymq |
| pkg_tinymq_fetch = git |
| pkg_tinymq_repo = https://github.com/ChicagoBoss/tinymq |
| pkg_tinymq_commit = master |
| |
| PACKAGES += tinymt |
| pkg_tinymt_name = tinymt |
| pkg_tinymt_description = TinyMT pseudo random number generator for Erlang. |
| pkg_tinymt_homepage = https://github.com/jj1bdx/tinymt-erlang |
| pkg_tinymt_fetch = git |
| pkg_tinymt_repo = https://github.com/jj1bdx/tinymt-erlang |
| pkg_tinymt_commit = master |
| |
| PACKAGES += tirerl |
| pkg_tirerl_name = tirerl |
| pkg_tirerl_description = Erlang interface to Elastic Search |
| pkg_tirerl_homepage = https://github.com/inaka/tirerl |
| pkg_tirerl_fetch = git |
| pkg_tirerl_repo = https://github.com/inaka/tirerl |
| pkg_tirerl_commit = master |
| |
| PACKAGES += toml |
| pkg_toml_name = toml |
| pkg_toml_description = TOML (0.4.0) config parser |
| pkg_toml_homepage = http://dozzie.jarowit.net/trac/wiki/TOML |
| pkg_toml_fetch = git |
| pkg_toml_repo = https://github.com/dozzie/toml |
| pkg_toml_commit = v0.2.0 |
| |
| PACKAGES += traffic_tools |
| pkg_traffic_tools_name = traffic_tools |
| pkg_traffic_tools_description = Simple traffic limiting library |
| pkg_traffic_tools_homepage = https://github.com/systra/traffic_tools |
| pkg_traffic_tools_fetch = git |
| pkg_traffic_tools_repo = https://github.com/systra/traffic_tools |
| pkg_traffic_tools_commit = master |
| |
| PACKAGES += trails |
| pkg_trails_name = trails |
| pkg_trails_description = A couple of improvements over Cowboy Routes |
| pkg_trails_homepage = http://inaka.github.io/cowboy-trails/ |
| pkg_trails_fetch = git |
| pkg_trails_repo = https://github.com/inaka/cowboy-trails |
| pkg_trails_commit = master |
| |
| PACKAGES += trane |
| pkg_trane_name = trane |
| pkg_trane_description = SAX style broken HTML parser in Erlang |
| pkg_trane_homepage = https://github.com/massemanet/trane |
| pkg_trane_fetch = git |
| pkg_trane_repo = https://github.com/massemanet/trane |
| pkg_trane_commit = master |
| |
| PACKAGES += trie |
| pkg_trie_name = trie |
| pkg_trie_description = Erlang Trie Implementation |
| pkg_trie_homepage = https://github.com/okeuday/trie |
| pkg_trie_fetch = git |
| pkg_trie_repo = https://github.com/okeuday/trie |
| pkg_trie_commit = master |
| |
| PACKAGES += triq |
| pkg_triq_name = triq |
| pkg_triq_description = Trifork QuickCheck |
| pkg_triq_homepage = https://triq.gitlab.io |
| pkg_triq_fetch = git |
| pkg_triq_repo = https://gitlab.com/triq/triq.git |
| pkg_triq_commit = master |
| |
| PACKAGES += tunctl |
| pkg_tunctl_name = tunctl |
| pkg_tunctl_description = Erlang TUN/TAP interface |
| pkg_tunctl_homepage = https://github.com/msantos/tunctl |
| pkg_tunctl_fetch = git |
| pkg_tunctl_repo = https://github.com/msantos/tunctl |
| pkg_tunctl_commit = master |
| |
| PACKAGES += unicorn |
| pkg_unicorn_name = unicorn |
| pkg_unicorn_description = Generic configuration server |
| pkg_unicorn_homepage = https://github.com/shizzard/unicorn |
| pkg_unicorn_fetch = git |
| pkg_unicorn_repo = https://github.com/shizzard/unicorn |
| pkg_unicorn_commit = master |
| |
| PACKAGES += unsplit |
| pkg_unsplit_name = unsplit |
| pkg_unsplit_description = Resolves conflicts in Mnesia after network splits |
| pkg_unsplit_homepage = https://github.com/uwiger/unsplit |
| pkg_unsplit_fetch = git |
| pkg_unsplit_repo = https://github.com/uwiger/unsplit |
| pkg_unsplit_commit = master |
| |
| PACKAGES += uuid |
| pkg_uuid_name = uuid |
| pkg_uuid_description = Erlang UUID Implementation |
| pkg_uuid_homepage = https://github.com/okeuday/uuid |
| pkg_uuid_fetch = git |
| pkg_uuid_repo = https://github.com/okeuday/uuid |
| pkg_uuid_commit = master |
| |
| PACKAGES += ux |
| pkg_ux_name = ux |
| pkg_ux_description = Unicode eXtention for Erlang (Strings, Collation) |
| pkg_ux_homepage = https://github.com/erlang-unicode/ux |
| pkg_ux_fetch = git |
| pkg_ux_repo = https://github.com/erlang-unicode/ux |
| pkg_ux_commit = master |
| |
| PACKAGES += verx |
| pkg_verx_name = verx |
| pkg_verx_description = Erlang implementation of the libvirtd remote protocol |
| pkg_verx_homepage = https://github.com/msantos/verx |
| pkg_verx_fetch = git |
| pkg_verx_repo = https://github.com/msantos/verx |
| pkg_verx_commit = master |
| |
| PACKAGES += vmq_bridge |
| pkg_vmq_bridge_name = vmq_bridge |
| pkg_vmq_bridge_description = Component of VerneMQ: A distributed MQTT message broker |
| pkg_vmq_bridge_homepage = https://verne.mq/ |
| pkg_vmq_bridge_fetch = git |
| pkg_vmq_bridge_repo = https://github.com/erlio/vmq_bridge |
| pkg_vmq_bridge_commit = master |
| |
| PACKAGES += vmstats |
| pkg_vmstats_name = vmstats |
| pkg_vmstats_description = tiny Erlang app that works in conjunction with statsderl in order to generate information on the Erlang VM for graphite logs. |
| pkg_vmstats_homepage = https://github.com/ferd/vmstats |
| pkg_vmstats_fetch = git |
| pkg_vmstats_repo = https://github.com/ferd/vmstats |
| pkg_vmstats_commit = master |
| |
| PACKAGES += walrus |
| pkg_walrus_name = walrus |
| pkg_walrus_description = Walrus - Mustache-like Templating |
| pkg_walrus_homepage = https://github.com/devinus/walrus |
| pkg_walrus_fetch = git |
| pkg_walrus_repo = https://github.com/devinus/walrus |
| pkg_walrus_commit = master |
| |
| PACKAGES += webmachine |
| pkg_webmachine_name = webmachine |
| pkg_webmachine_description = A REST-based system for building web applications. |
| pkg_webmachine_homepage = https://github.com/basho/webmachine |
| pkg_webmachine_fetch = git |
| pkg_webmachine_repo = https://github.com/basho/webmachine |
| pkg_webmachine_commit = master |
| |
| PACKAGES += websocket_client |
| pkg_websocket_client_name = websocket_client |
| pkg_websocket_client_description = Erlang websocket client (ws and wss supported) |
| pkg_websocket_client_homepage = https://github.com/jeremyong/websocket_client |
| pkg_websocket_client_fetch = git |
| pkg_websocket_client_repo = https://github.com/jeremyong/websocket_client |
| pkg_websocket_client_commit = master |
| |
| PACKAGES += worker_pool |
| pkg_worker_pool_name = worker_pool |
| pkg_worker_pool_description = a simple erlang worker pool |
| pkg_worker_pool_homepage = https://github.com/inaka/worker_pool |
| pkg_worker_pool_fetch = git |
| pkg_worker_pool_repo = https://github.com/inaka/worker_pool |
| pkg_worker_pool_commit = main |
| |
| PACKAGES += wrangler |
| pkg_wrangler_name = wrangler |
| pkg_wrangler_description = Import of the Wrangler svn repository. |
| pkg_wrangler_homepage = http://www.cs.kent.ac.uk/projects/wrangler/Home.html |
| pkg_wrangler_fetch = git |
| pkg_wrangler_repo = https://github.com/RefactoringTools/wrangler |
| pkg_wrangler_commit = master |
| |
| PACKAGES += wsock |
| pkg_wsock_name = wsock |
| pkg_wsock_description = Erlang library to build WebSocket clients and servers |
| pkg_wsock_homepage = https://github.com/madtrick/wsock |
| pkg_wsock_fetch = git |
| pkg_wsock_repo = https://github.com/madtrick/wsock |
| pkg_wsock_commit = master |
| |
| PACKAGES += xhttpc |
| pkg_xhttpc_name = xhttpc |
| pkg_xhttpc_description = Extensible HTTP Client for Erlang |
| pkg_xhttpc_homepage = https://github.com/seriyps/xhttpc |
| pkg_xhttpc_fetch = git |
| pkg_xhttpc_repo = https://github.com/seriyps/xhttpc |
| pkg_xhttpc_commit = master |
| |
| PACKAGES += xref_runner |
| pkg_xref_runner_name = xref_runner |
| pkg_xref_runner_description = Erlang Xref Runner (inspired in rebar xref) |
| pkg_xref_runner_homepage = https://github.com/inaka/xref_runner |
| pkg_xref_runner_fetch = git |
| pkg_xref_runner_repo = https://github.com/inaka/xref_runner |
| pkg_xref_runner_commit = master |
| |
| PACKAGES += yamerl |
| pkg_yamerl_name = yamerl |
| pkg_yamerl_description = YAML 1.2 parser in pure Erlang |
| pkg_yamerl_homepage = https://github.com/yakaz/yamerl |
| pkg_yamerl_fetch = git |
| pkg_yamerl_repo = https://github.com/yakaz/yamerl |
| pkg_yamerl_commit = master |
| |
| PACKAGES += yamler |
| pkg_yamler_name = yamler |
| pkg_yamler_description = libyaml-based yaml loader for Erlang |
| pkg_yamler_homepage = https://github.com/goertzenator/yamler |
| pkg_yamler_fetch = git |
| pkg_yamler_repo = https://github.com/goertzenator/yamler |
| pkg_yamler_commit = master |
| |
| PACKAGES += yaws |
| pkg_yaws_name = yaws |
| pkg_yaws_description = Yaws webserver |
| pkg_yaws_homepage = http://yaws.hyber.org |
| pkg_yaws_fetch = git |
| pkg_yaws_repo = https://github.com/klacke/yaws |
| pkg_yaws_commit = master |
| |
| PACKAGES += zippers |
| pkg_zippers_name = zippers |
| pkg_zippers_description = A library for functional zipper data structures in Erlang. Read more on zippers |
| pkg_zippers_homepage = https://github.com/ferd/zippers |
| pkg_zippers_fetch = git |
| pkg_zippers_repo = https://github.com/ferd/zippers |
| pkg_zippers_commit = master |
| |
| PACKAGES += zlists |
| pkg_zlists_name = zlists |
| pkg_zlists_description = Erlang lazy lists library. |
| pkg_zlists_homepage = https://github.com/vjache/erlang-zlists |
| pkg_zlists_fetch = git |
| pkg_zlists_repo = https://github.com/vjache/erlang-zlists |
| pkg_zlists_commit = master |
| |
| PACKAGES += zucchini |
| pkg_zucchini_name = zucchini |
| pkg_zucchini_description = An Erlang INI parser |
| pkg_zucchini_homepage = https://github.com/devinus/zucchini |
| pkg_zucchini_fetch = git |
| pkg_zucchini_repo = https://github.com/devinus/zucchini |
| pkg_zucchini_commit = master |
| |
| # Copyright (c) 2015-2016, Loïc Hoguin <essen@ninenines.eu> |
| # This file is part of erlang.mk and subject to the terms of the ISC License. |
| |
| .PHONY: search |
| |
| define pkg_print |
| $(verbose) printf "%s\n" \ |
| $(if $(call core_eq,$(1),$(pkg_$(1)_name)),,"Pkg name: $(1)") \ |
| "App name: $(pkg_$(1)_name)" \ |
| "Description: $(pkg_$(1)_description)" \ |
| "Home page: $(pkg_$(1)_homepage)" \ |
| "Fetch with: $(pkg_$(1)_fetch)" \ |
| "Repository: $(pkg_$(1)_repo)" \ |
| "Commit: $(pkg_$(1)_commit)" \ |
| "" |
| |
| endef |
| |
| search: |
| ifdef q |
| $(foreach p,$(PACKAGES), \ |
| $(if $(findstring $(call core_lc,$(q)),$(call core_lc,$(pkg_$(p)_name) $(pkg_$(p)_description))), \ |
| $(call pkg_print,$(p)))) |
| else |
| $(foreach p,$(PACKAGES),$(call pkg_print,$(p))) |
| endif |
| |
| # Copyright (c) 2013-2016, Loïc Hoguin <essen@ninenines.eu> |
| # This file is part of erlang.mk and subject to the terms of the ISC License. |
| |
| .PHONY: distclean-deps clean-tmp-deps.log |
| |
| # Configuration. |
| |
| ifdef OTP_DEPS |
| $(warning The variable OTP_DEPS is deprecated in favor of LOCAL_DEPS.) |
| endif |
| |
| IGNORE_DEPS ?= |
| export IGNORE_DEPS |
| |
| APPS_DIR ?= $(CURDIR)/apps |
| export APPS_DIR |
| |
| DEPS_DIR ?= $(CURDIR)/deps |
| export DEPS_DIR |
| |
| REBAR_DEPS_DIR = $(DEPS_DIR) |
| export REBAR_DEPS_DIR |
| |
| # When testing Erlang.mk and updating these, make sure |
| # to delete test/test_rebar_git before running tests again. |
| REBAR3_GIT ?= https://github.com/erlang/rebar3 |
| REBAR3_COMMIT ?= bde4b54248d16280b2c70a244aca3bb7566e2033 # 3.23.0 |
| |
| CACHE_DEPS ?= 0 |
| |
| CACHE_DIR ?= $(if $(XDG_CACHE_HOME),$(XDG_CACHE_HOME),$(HOME)/.cache)/erlang.mk |
| export CACHE_DIR |
| |
| # External "early" plugins (see core/plugins.mk for regular plugins). |
| # They both use the core_dep_plugin macro. |
| |
| define core_dep_plugin |
| ifeq ($(2),$(PROJECT)) |
| -include $$(patsubst $(PROJECT)/%,%,$(1)) |
| else |
| -include $(DEPS_DIR)/$(1) |
| |
| $(DEPS_DIR)/$(1): $(DEPS_DIR)/$(2) ; |
| endif |
| endef |
| |
| DEP_EARLY_PLUGINS ?= |
| |
| $(foreach p,$(DEP_EARLY_PLUGINS),\ |
| $(eval $(if $(findstring /,$p),\ |
| $(call core_dep_plugin,$p,$(firstword $(subst /, ,$p))),\ |
| $(call core_dep_plugin,$p/early-plugins.mk,$p)))) |
| |
| # Query functions. |
| |
| query_fetch_method = $(if $(dep_$(1)),$(call _qfm_dep,$(word 1,$(dep_$(1)))),$(call _qfm_pkg,$(1))) |
| _qfm_dep = $(if $(dep_fetch_$(1)),$(1),$(if $(IS_DEP),legacy,fail)) |
| _qfm_pkg = $(if $(pkg_$(1)_fetch),$(pkg_$(1)_fetch),fail) |
| |
| query_name = $(if $(dep_$(1)),$(1),$(if $(pkg_$(1)_name),$(pkg_$(1)_name),$(1))) |
| |
| query_repo = $(call _qr,$(1),$(call query_fetch_method,$(1))) |
| _qr = $(if $(query_repo_$(2)),$(call query_repo_$(2),$(1)),$(call dep_repo,$(1))) |
| |
| query_repo_default = $(if $(dep_$(1)),$(word 2,$(dep_$(1))),$(pkg_$(1)_repo)) |
| query_repo_git = $(patsubst git://github.com/%,https://github.com/%,$(call query_repo_default,$(1))) |
| query_repo_git-subfolder = $(call query_repo_git,$(1)) |
| query_repo_git-submodule = - |
| query_repo_hg = $(call query_repo_default,$(1)) |
| query_repo_svn = $(call query_repo_default,$(1)) |
| query_repo_cp = $(call query_repo_default,$(1)) |
| query_repo_ln = $(call query_repo_default,$(1)) |
| query_repo_hex = https://hex.pm/packages/$(if $(word 3,$(dep_$(1))),$(word 3,$(dep_$(1))),$(1)) |
| query_repo_fail = - |
| query_repo_legacy = - |
| |
| query_version = $(call _qv,$(1),$(call query_fetch_method,$(1))) |
| _qv = $(if $(query_version_$(2)),$(call query_version_$(2),$(1)),$(call dep_commit,$(1))) |
| |
| query_version_default = $(if $(dep_$(1)_commit),$(dep_$(1)_commit),$(if $(dep_$(1)),$(word 3,$(dep_$(1))),$(pkg_$(1)_commit))) |
| query_version_git = $(call query_version_default,$(1)) |
| query_version_git-subfolder = $(call query_version_git,$(1)) |
| query_version_git-submodule = - |
| query_version_hg = $(call query_version_default,$(1)) |
| query_version_svn = - |
| query_version_cp = - |
| query_version_ln = - |
| query_version_hex = $(if $(dep_$(1)_commit),$(dep_$(1)_commit),$(if $(dep_$(1)),$(word 2,$(dep_$(1))),$(pkg_$(1)_commit))) |
| query_version_fail = - |
| query_version_legacy = - |
| |
| query_extra = $(call _qe,$(1),$(call query_fetch_method,$(1))) |
| _qe = $(if $(query_extra_$(2)),$(call query_extra_$(2),$(1)),-) |
| |
| query_extra_git = - |
| query_extra_git-subfolder = $(if $(dep_$(1)),subfolder=$(word 4,$(dep_$(1))),-) |
| query_extra_git-submodule = - |
| query_extra_hg = - |
| query_extra_svn = - |
| query_extra_cp = - |
| query_extra_ln = - |
| query_extra_hex = $(if $(dep_$(1)),package-name=$(word 3,$(dep_$(1))),-) |
| query_extra_fail = - |
| query_extra_legacy = - |
| |
| query_absolute_path = $(addprefix $(DEPS_DIR)/,$(call query_name,$(1))) |
| |
| # Deprecated legacy query functions. |
| dep_fetch = $(call query_fetch_method,$(1)) |
| dep_name = $(call query_name,$(1)) |
| dep_repo = $(call query_repo_git,$(1)) |
| dep_commit = $(if $(dep_$(1)_commit),$(dep_$(1)_commit),$(if $(dep_$(1)),$(if $(filter hex,$(word 1,$(dep_$(1)))),$(word 2,$(dep_$(1))),$(word 3,$(dep_$(1)))),$(pkg_$(1)_commit))) |
| |
| LOCAL_DEPS_DIRS = $(foreach a,$(LOCAL_DEPS),$(if $(wildcard $(APPS_DIR)/$(a)),$(APPS_DIR)/$(a))) |
| ALL_DEPS_DIRS = $(addprefix $(DEPS_DIR)/,$(foreach dep,$(filter-out $(IGNORE_DEPS),$(BUILD_DEPS) $(DEPS)),$(call dep_name,$(dep)))) |
| |
| # When we are calling an app directly we don't want to include it here |
| # otherwise it'll be treated both as an apps and a top-level project. |
| ALL_APPS_DIRS = $(if $(wildcard $(APPS_DIR)/),$(filter-out $(APPS_DIR),$(shell find $(APPS_DIR) -maxdepth 1 -type d))) |
| ifdef ROOT_DIR |
| ifndef IS_APP |
| ALL_APPS_DIRS := $(filter-out $(APPS_DIR)/$(notdir $(CURDIR)),$(ALL_APPS_DIRS)) |
| endif |
| endif |
| |
| ifeq ($(filter $(APPS_DIR) $(DEPS_DIR),$(subst :, ,$(ERL_LIBS))),) |
| ifeq ($(ERL_LIBS),) |
| ERL_LIBS = $(APPS_DIR):$(DEPS_DIR) |
| else |
| ERL_LIBS := $(ERL_LIBS):$(APPS_DIR):$(DEPS_DIR) |
| endif |
| endif |
| export ERL_LIBS |
| |
| export NO_AUTOPATCH |
| |
| # Verbosity. |
| |
| dep_verbose_0 = @echo " DEP $1 ($(call dep_commit,$1))"; |
| dep_verbose_2 = set -x; |
| dep_verbose = $(dep_verbose_$(V)) |
| |
| # Optimization: don't recompile deps unless truly necessary. |
| |
| ifndef IS_DEP |
| ifneq ($(MAKELEVEL),0) |
| $(shell rm -f ebin/dep_built) |
| endif |
| endif |
| |
| # Core targets. |
| |
| ALL_APPS_DIRS_TO_BUILD = $(if $(LOCAL_DEPS_DIRS)$(IS_APP),$(LOCAL_DEPS_DIRS),$(ALL_APPS_DIRS)) |
| |
| apps:: $(ALL_APPS_DIRS) clean-tmp-deps.log | $(ERLANG_MK_TMP) |
| # Create ebin directory for all apps to make sure Erlang recognizes them |
| # as proper OTP applications when using -include_lib. This is a temporary |
| # fix, a proper fix would be to compile apps/* in the right order. |
| ifndef IS_APP |
| ifneq ($(ALL_APPS_DIRS),) |
| $(verbose) set -e; for dep in $(ALL_APPS_DIRS) ; do \ |
| mkdir -p $$dep/ebin; \ |
| done |
| endif |
| endif |
| # At the toplevel: if LOCAL_DEPS is defined with at least one local app, only |
| # compile that list of apps. Otherwise, compile everything. |
| # Within an app: compile all LOCAL_DEPS that are (uncompiled) local apps. |
| ifneq ($(ALL_APPS_DIRS_TO_BUILD),) |
| $(verbose) set -e; for dep in $(ALL_APPS_DIRS_TO_BUILD); do \ |
| if grep -qs ^$$dep$$ $(ERLANG_MK_TMP)/apps.log; then \ |
| :; \ |
| else \ |
| echo $$dep >> $(ERLANG_MK_TMP)/apps.log; \ |
| $(MAKE) -C $$dep $(if $(IS_TEST),test-build-app) IS_APP=1; \ |
| fi \ |
| done |
| endif |
| |
| clean-tmp-deps.log: |
| ifeq ($(IS_APP)$(IS_DEP),) |
| $(verbose) rm -f $(ERLANG_MK_TMP)/apps.log $(ERLANG_MK_TMP)/deps.log |
| endif |
| |
| # Erlang.mk does not rebuild dependencies after they were compiled |
| # once. If a developer is working on the top-level project and some |
| # dependencies at the same time, he may want to change this behavior. |
| # There are two solutions: |
| # 1. Set `FULL=1` so that all dependencies are visited and |
| # recursively recompiled if necessary. |
| # 2. Set `FORCE_REBUILD=` to the specific list of dependencies that |
| # should be recompiled (instead of the whole set). |
| |
| FORCE_REBUILD ?= |
| |
| ifeq ($(origin FULL),undefined) |
| ifneq ($(strip $(force_rebuild_dep)$(FORCE_REBUILD)),) |
| define force_rebuild_dep |
| echo "$(FORCE_REBUILD)" | grep -qw "$$(basename "$1")" |
| endef |
| endif |
| endif |
| |
| ifneq ($(SKIP_DEPS),) |
| deps:: |
| else |
| deps:: $(ALL_DEPS_DIRS) apps clean-tmp-deps.log | $(ERLANG_MK_TMP) |
| ifneq ($(ALL_DEPS_DIRS),) |
| $(verbose) set -e; for dep in $(ALL_DEPS_DIRS); do \ |
| if grep -qs ^$$dep$$ $(ERLANG_MK_TMP)/deps.log; then \ |
| :; \ |
| else \ |
| echo $$dep >> $(ERLANG_MK_TMP)/deps.log; \ |
| if [ -z "$(strip $(FULL))" ] $(if $(force_rebuild_dep),&& ! ($(call force_rebuild_dep,$$dep)),) && [ ! -L $$dep ] && [ -f $$dep/ebin/dep_built ]; then \ |
| :; \ |
| elif [ "$$dep" = "$(DEPS_DIR)/hut" -a "$(HUT_PATCH)" ]; then \ |
| $(MAKE) -C $$dep app IS_DEP=1; \ |
| if [ ! -L $$dep ] && [ -d $$dep/ebin ]; then touch $$dep/ebin/dep_built; fi; \ |
| elif [ -f $$dep/GNUmakefile ] || [ -f $$dep/makefile ] || [ -f $$dep/Makefile ]; then \ |
| $(MAKE) -C $$dep IS_DEP=1; \ |
| if [ ! -L $$dep ] && [ -d $$dep/ebin ]; then touch $$dep/ebin/dep_built; fi; \ |
| else \ |
| echo "Error: No Makefile to build dependency $$dep." >&2; \ |
| exit 2; \ |
| fi \ |
| fi \ |
| done |
| endif |
| endif |
| |
| # Deps related targets. |
| |
| # @todo rename GNUmakefile and makefile into Makefile first, if they exist |
| # While Makefile file could be GNUmakefile or makefile, |
| # in practice only Makefile is needed so far. |
| define dep_autopatch |
| if [ -f $(DEPS_DIR)/$(1)/erlang.mk ]; then \ |
| rm -rf $(DEPS_DIR)/$1/ebin/; \ |
| $(call erlang,$(call dep_autopatch_appsrc.erl,$(1))); \ |
| $(call dep_autopatch_erlang_mk,$(1)); \ |
| elif [ -f $(DEPS_DIR)/$(1)/Makefile ]; then \ |
| if [ -f $(DEPS_DIR)/$1/rebar.lock ]; then \ |
| $(call dep_autopatch2,$1); \ |
| elif [ 0 != `grep -c "include ../\w*\.mk" $(DEPS_DIR)/$(1)/Makefile` ]; then \ |
| $(call dep_autopatch2,$(1)); \ |
| elif [ 0 != `grep -ci "^[^#].*rebar" $(DEPS_DIR)/$(1)/Makefile` ]; then \ |
| $(call dep_autopatch2,$(1)); \ |
| elif [ -n "`find $(DEPS_DIR)/$(1)/ -type f -name \*.mk -not -name erlang.mk -exec grep -i "^[^#].*rebar" '{}' \;`" ]; then \ |
| $(call dep_autopatch2,$(1)); \ |
| fi \ |
| else \ |
| if [ ! -d $(DEPS_DIR)/$(1)/src/ ]; then \ |
| $(call dep_autopatch_noop,$(1)); \ |
| else \ |
| $(call dep_autopatch2,$(1)); \ |
| fi \ |
| fi |
| endef |
| |
| define dep_autopatch2 |
| ! test -f $(DEPS_DIR)/$1/ebin/$1.app || \ |
| mv -n $(DEPS_DIR)/$1/ebin/$1.app $(DEPS_DIR)/$1/src/$1.app.src; \ |
| rm -f $(DEPS_DIR)/$1/ebin/$1.app; \ |
| if [ -f $(DEPS_DIR)/$1/src/$1.app.src.script ]; then \ |
| $(call erlang,$(call dep_autopatch_appsrc_script.erl,$(1))); \ |
| fi; \ |
| $(call erlang,$(call dep_autopatch_appsrc.erl,$(1))); \ |
| if [ -f $(DEPS_DIR)/$(1)/rebar -o -f $(DEPS_DIR)/$(1)/rebar.config -o -f $(DEPS_DIR)/$(1)/rebar.config.script -o -f $(DEPS_DIR)/$1/rebar.lock ]; then \ |
| $(call dep_autopatch_fetch_rebar); \ |
| $(call dep_autopatch_rebar,$(1)); \ |
| else \ |
| $(call dep_autopatch_gen,$(1)); \ |
| fi |
| endef |
| |
| define dep_autopatch_noop |
| printf "noop:\n" > $(DEPS_DIR)/$(1)/Makefile |
| endef |
| |
| # Replace "include erlang.mk" with a line that will load the parent Erlang.mk |
| # if given. Do it for all 3 possible Makefile file names. |
| ifeq ($(NO_AUTOPATCH_ERLANG_MK),) |
| define dep_autopatch_erlang_mk |
| for f in Makefile makefile GNUmakefile; do \ |
| if [ -f $(DEPS_DIR)/$1/$$f ]; then \ |
| sed -i.bak s/'include *erlang.mk'/'include $$(if $$(ERLANG_MK_FILENAME),$$(ERLANG_MK_FILENAME),erlang.mk)'/ $(DEPS_DIR)/$1/$$f; \ |
| fi \ |
| done |
| endef |
| else |
| define dep_autopatch_erlang_mk |
| : |
| endef |
| endif |
| |
| define dep_autopatch_gen |
| printf "%s\n" \ |
| "ERLC_OPTS = +debug_info" \ |
| "include ../../erlang.mk" > $(DEPS_DIR)/$(1)/Makefile |
| endef |
| |
| # We use flock/lockf when available to avoid concurrency issues. |
| define dep_autopatch_fetch_rebar |
| if command -v flock >/dev/null; then \ |
| flock $(ERLANG_MK_TMP)/rebar.lock sh -c "$(call dep_autopatch_fetch_rebar2)"; \ |
| elif command -v lockf >/dev/null; then \ |
| lockf $(ERLANG_MK_TMP)/rebar.lock sh -c "$(call dep_autopatch_fetch_rebar2)"; \ |
| else \ |
| $(call dep_autopatch_fetch_rebar2); \ |
| fi |
| endef |
| |
| define dep_autopatch_fetch_rebar2 |
| if [ ! -d $(ERLANG_MK_TMP)/rebar3 ]; then \ |
| git clone -q -n -- $(REBAR3_GIT) $(ERLANG_MK_TMP)/rebar3; \ |
| cd $(ERLANG_MK_TMP)/rebar3; \ |
| git checkout -q $(REBAR3_COMMIT); \ |
| ./bootstrap; \ |
| cd -; \ |
| fi |
| endef |
| |
| define dep_autopatch_rebar |
| if [ -f $(DEPS_DIR)/$(1)/Makefile ]; then \ |
| mv $(DEPS_DIR)/$(1)/Makefile $(DEPS_DIR)/$(1)/Makefile.orig.mk; \ |
| fi; \ |
| $(call erlang,$(call dep_autopatch_rebar.erl,$(1))); \ |
| rm -f $(DEPS_DIR)/$(1)/ebin/$(1).app |
| endef |
| |
| define dep_autopatch_rebar.erl |
| application:load(rebar), |
| application:set_env(rebar, log_level, debug), |
| {module, rebar3} = c:l(rebar3), |
| Conf1 = case file:consult("$(call core_native_path,$(DEPS_DIR)/$1/rebar.config)") of |
| {ok, Conf0} -> Conf0; |
| _ -> [] |
| end, |
| {Conf, OsEnv} = fun() -> |
| case filelib:is_file("$(call core_native_path,$(DEPS_DIR)/$1/rebar.config.script)") of |
| false -> {Conf1, []}; |
| true -> |
| Bindings0 = erl_eval:new_bindings(), |
| Bindings1 = erl_eval:add_binding('CONFIG', Conf1, Bindings0), |
| Bindings = erl_eval:add_binding('SCRIPT', "$(call core_native_path,$(DEPS_DIR)/$1/rebar.config.script)", Bindings1), |
| Before = os:getenv(), |
| {ok, Conf2} = file:script("$(call core_native_path,$(DEPS_DIR)/$1/rebar.config.script)", Bindings), |
| {Conf2, lists:foldl(fun(E, Acc) -> lists:delete(E, Acc) end, os:getenv(), Before)} |
| end |
| end(), |
| Write = fun (Text) -> |
| file:write_file("$(call core_native_path,$(DEPS_DIR)/$1/Makefile)", Text, [append]) |
| end, |
| Escape = fun (Text) -> |
| re:replace(Text, "\\\\$$", "\$$$$", [global, {return, list}]) |
| end, |
| Write("IGNORE_DEPS += edown eper eunit_formatters meck node_package " |
| "rebar_lock_deps_plugin rebar_vsn_plugin reltool_util\n"), |
| Write("C_SRC_DIR = /path/do/not/exist\n"), |
| Write("C_SRC_TYPE = rebar\n"), |
| Write("DRV_CFLAGS = -fPIC\nexport DRV_CFLAGS\n"), |
| Write(["ERLANG_ARCH = ", rebar_utils:wordsize(), "\nexport ERLANG_ARCH\n"]), |
| ToList = fun |
| (V) when is_atom(V) -> atom_to_list(V); |
| (V) when is_list(V) -> "'\\"" ++ V ++ "\\"'" |
| end, |
| fun() -> |
| Write("ERLC_OPTS = +debug_info\n"), |
| case lists:keyfind(erl_opts, 1, Conf) of |
| false -> ok; |
| {_, ErlOpts} -> |
| lists:foreach(fun |
| ({d, D}) -> |
| Write("ERLC_OPTS += -D" ++ ToList(D) ++ "=1\n"); |
| ({d, DKey, DVal}) -> |
| Write("ERLC_OPTS += -D" ++ ToList(DKey) ++ "=" ++ ToList(DVal) ++ "\n"); |
| ({i, I}) -> |
| Write(["ERLC_OPTS += -I ", I, "\n"]); |
| ({platform_define, Regex, D}) -> |
| case rebar_utils:is_arch(Regex) of |
| true -> Write("ERLC_OPTS += -D" ++ ToList(D) ++ "=1\n"); |
| false -> ok |
| end; |
| ({parse_transform, PT}) -> |
| Write("ERLC_OPTS += +'{parse_transform, " ++ ToList(PT) ++ "}'\n"); |
| (_) -> ok |
| end, ErlOpts) |
| end, |
| Write("\n") |
| end(), |
| GetHexVsn2 = fun(N, NP) -> |
| case file:consult("$(call core_native_path,$(DEPS_DIR)/$1/rebar.lock)") of |
| {ok, Lock} -> |
| io:format("~p~n", [Lock]), |
| LockPkgs = case lists:keyfind("1.2.0", 1, Lock) of |
| {_, LP} -> |
| LP; |
| _ -> |
| case lists:keyfind("1.1.0", 1, Lock) of |
| {_, LP} -> |
| LP; |
| _ -> |
| false |
| end |
| end, |
| if |
| is_list(LockPkgs) -> |
| io:format("~p~n", [LockPkgs]), |
| case lists:keyfind(atom_to_binary(N, latin1), 1, LockPkgs) of |
| {_, {pkg, _, Vsn}, _} -> |
| io:format("~p~n", [Vsn]), |
| {N, {hex, NP, binary_to_list(Vsn)}}; |
| _ -> |
| false |
| end; |
| true -> |
| false |
| end; |
| _ -> |
| false |
| end |
| end, |
| GetHexVsn3Common = fun(N, NP, S0) -> |
| case GetHexVsn2(N, NP) of |
| false -> |
| S2 = case S0 of |
| " " ++ S1 -> S1; |
| _ -> S0 |
| end, |
| S = case length([ok || $$. <- S2]) of |
| 0 -> S2 ++ ".0.0"; |
| 1 -> S2 ++ ".0"; |
| _ -> S2 |
| end, |
| {N, {hex, NP, S}}; |
| NameSource -> |
| NameSource |
| end |
| end, |
| GetHexVsn3 = fun |
| (N, NP, "~>" ++ S0) -> |
| GetHexVsn3Common(N, NP, S0); |
| (N, NP, ">=" ++ S0) -> |
| GetHexVsn3Common(N, NP, S0); |
| (N, NP, S) -> {N, {hex, NP, S}} |
| end, |
| fun() -> |
| File = case lists:keyfind(deps, 1, Conf) of |
| false -> []; |
| {_, Deps} -> |
| [begin case case Dep of |
| N when is_atom(N) -> GetHexVsn2(N, N); |
| {N, S} when is_atom(N), is_list(S) -> GetHexVsn3(N, N, S); |
| {N, {pkg, NP}} when is_atom(N) -> GetHexVsn2(N, NP); |
| {N, S, {pkg, NP}} -> GetHexVsn3(N, NP, S); |
| {N, S} when is_tuple(S) -> {N, S}; |
| {N, _, S} -> {N, S}; |
| {N, _, S, _} -> {N, S}; |
| _ -> false |
| end of |
| false -> ok; |
| {Name, Source} -> |
| {Method, Repo, Commit} = case Source of |
| {hex, NPV, V} -> {hex, V, NPV}; |
| {git, R} -> {git, R, master}; |
| {M, R, {branch, C}} -> {M, R, C}; |
| {M, R, {ref, C}} -> {M, R, C}; |
| {M, R, {tag, C}} -> {M, R, C}; |
| {M, R, C} -> {M, R, C} |
| end, |
| Write(io_lib:format("DEPS += ~s\ndep_~s = ~s ~s ~s~n", [Name, Name, Method, Repo, Commit])) |
| end end || Dep <- Deps] |
| end |
| end(), |
| fun() -> |
| case lists:keyfind(erl_first_files, 1, Conf) of |
| false -> ok; |
| {_, Files0} -> |
| Files = [begin |
| hd(filelib:wildcard("$(call core_native_path,$(DEPS_DIR)/$1/src/)**/" ++ filename:rootname(F) ++ ".*rl")) |
| end || "src/" ++ F <- Files0], |
| Names = [[" ", case lists:reverse(F) of |
| "lre." ++ Elif -> lists:reverse(Elif); |
| "lrx." ++ Elif -> lists:reverse(Elif); |
| "lry." ++ Elif -> lists:reverse(Elif); |
| Elif -> lists:reverse(Elif) |
| end] || "$(call core_native_path,$(DEPS_DIR)/$1/src/)" ++ F <- Files], |
| Write(io_lib:format("COMPILE_FIRST +=~s\n", [Names])) |
| end |
| end(), |
| Write("\n\nrebar_dep: preprocess pre-deps deps pre-app app\n"), |
| Write("\npreprocess::\n"), |
| Write("\npre-deps::\n"), |
| Write("\npre-app::\n"), |
| PatchHook = fun(Cmd) -> |
| Cmd2 = re:replace(Cmd, "^([g]?make)(.*)( -C.*)", "\\\\1\\\\3\\\\2", [{return, list}]), |
| case Cmd2 of |
| "make -C" ++ Cmd1 -> "$$\(MAKE) -C" ++ Escape(Cmd1); |
| "gmake -C" ++ Cmd1 -> "$$\(MAKE) -C" ++ Escape(Cmd1); |
| "make " ++ Cmd1 -> "$$\(MAKE) -f Makefile.orig.mk " ++ Escape(Cmd1); |
| "gmake " ++ Cmd1 -> "$$\(MAKE) -f Makefile.orig.mk " ++ Escape(Cmd1); |
| _ -> Escape(Cmd) |
| end |
| end, |
| fun() -> |
| case lists:keyfind(pre_hooks, 1, Conf) of |
| false -> ok; |
| {_, Hooks} -> |
| [case H of |
| {'get-deps', Cmd} -> |
| Write("\npre-deps::\n\t" ++ PatchHook(Cmd) ++ "\n"); |
| {compile, Cmd} -> |
| Write("\npre-app::\n\tCC=$$\(CC) " ++ PatchHook(Cmd) ++ "\n"); |
| {{pc, compile}, Cmd} -> |
| Write("\npre-app::\n\tCC=$$\(CC) " ++ PatchHook(Cmd) ++ "\n"); |
| {Regex, compile, Cmd} -> |
| case rebar_utils:is_arch(Regex) of |
| true -> Write("\npre-app::\n\tCC=$$\(CC) " ++ PatchHook(Cmd) ++ "\n"); |
| false -> ok |
| end; |
| _ -> ok |
| end || H <- Hooks] |
| end |
| end(), |
| ShellToMk = fun(V0) -> |
| V1 = re:replace(V0, "[$$][(]", "$$\(shell ", [global]), |
| V = re:replace(V1, "([$$])(?![(])(\\\\w*)", "\\\\1(\\\\2)", [global]), |
| re:replace(V, "-Werror\\\\b", "", [{return, list}, global]) |
| end, |
| PortSpecs = fun() -> |
| case lists:keyfind(port_specs, 1, Conf) of |
| false -> |
| case filelib:is_dir("$(call core_native_path,$(DEPS_DIR)/$1/c_src)") of |
| false -> []; |
| true -> |
| [{"priv/" ++ proplists:get_value(so_name, Conf, "$(1)_drv.so"), |
| proplists:get_value(port_sources, Conf, ["c_src/*.c"]), []}] |
| end; |
| {_, Specs} -> |
| lists:flatten([case S of |
| {Output, Input} -> {ShellToMk(Output), Input, []}; |
| {Regex, Output, Input} -> |
| case rebar_utils:is_arch(Regex) of |
| true -> {ShellToMk(Output), Input, []}; |
| false -> [] |
| end; |
| {Regex, Output, Input, [{env, Env}]} -> |
| case rebar_utils:is_arch(Regex) of |
| true -> {ShellToMk(Output), Input, Env}; |
| false -> [] |
| end |
| end || S <- Specs]) |
| end |
| end(), |
| PortSpecWrite = fun (Text) -> |
| file:write_file("$(call core_native_path,$(DEPS_DIR)/$1/c_src/Makefile.erlang.mk)", Text, [append]) |
| end, |
| case PortSpecs of |
| [] -> ok; |
| _ -> |
| Write("\npre-app::\n\t@$$\(MAKE) --no-print-directory -f c_src/Makefile.erlang.mk\n"), |
| PortSpecWrite(io_lib:format("ERL_CFLAGS ?= -finline-functions -Wall -fPIC -I \\"~s/erts-~s/include\\" -I \\"~s\\"\n", |
| [code:root_dir(), erlang:system_info(version), code:lib_dir(erl_interface, include)])), |
| PortSpecWrite(io_lib:format("ERL_LDFLAGS ?= -L \\"~s\\" -lei\n", |
| [code:lib_dir(erl_interface, lib)])), |
| [PortSpecWrite(["\n", E, "\n"]) || E <- OsEnv], |
| FilterEnv = fun(Env) -> |
| lists:flatten([case E of |
| {_, _} -> E; |
| {Regex, K, V} -> |
| case rebar_utils:is_arch(Regex) of |
| true -> {K, V}; |
| false -> [] |
| end |
| end || E <- Env]) |
| end, |
| MergeEnv = fun(Env) -> |
| lists:foldl(fun ({K, V}, Acc) -> |
| case lists:keyfind(K, 1, Acc) of |
| false -> [{K, rebar_utils:expand_env_variable(V, K, "")}|Acc]; |
| {_, V0} -> [{K, rebar_utils:expand_env_variable(V, K, V0)}|Acc] |
| end |
| end, [], Env) |
| end, |
| PortEnv = case lists:keyfind(port_env, 1, Conf) of |
| false -> []; |
| {_, PortEnv0} -> FilterEnv(PortEnv0) |
| end, |
| PortSpec = fun ({Output, Input0, Env}) -> |
| filelib:ensure_dir("$(call core_native_path,$(DEPS_DIR)/$1/)" ++ Output), |
| Input = [[" ", I] || I <- Input0], |
| PortSpecWrite([ |
| [["\n", K, " = ", ShellToMk(V)] || {K, V} <- lists:reverse(MergeEnv(PortEnv))], |
| case $(PLATFORM) of |
| darwin -> "\n\nLDFLAGS += -flat_namespace -undefined suppress"; |
| _ -> "" |
| end, |
| "\n\nall:: ", Output, "\n\t@:\n\n", |
| "%.o: %.c\n\t$$\(CC) -c -o $$\@ $$\< $$\(CFLAGS) $$\(ERL_CFLAGS) $$\(DRV_CFLAGS) $$\(EXE_CFLAGS)\n\n", |
| "%.o: %.C\n\t$$\(CXX) -c -o $$\@ $$\< $$\(CXXFLAGS) $$\(ERL_CFLAGS) $$\(DRV_CFLAGS) $$\(EXE_CFLAGS)\n\n", |
| "%.o: %.cc\n\t$$\(CXX) -c -o $$\@ $$\< $$\(CXXFLAGS) $$\(ERL_CFLAGS) $$\(DRV_CFLAGS) $$\(EXE_CFLAGS)\n\n", |
| "%.o: %.cpp\n\t$$\(CXX) -c -o $$\@ $$\< $$\(CXXFLAGS) $$\(ERL_CFLAGS) $$\(DRV_CFLAGS) $$\(EXE_CFLAGS)\n\n", |
| [[Output, ": ", K, " += ", ShellToMk(V), "\n"] || {K, V} <- lists:reverse(MergeEnv(FilterEnv(Env)))], |
| Output, ": $$\(foreach ext,.c .C .cc .cpp,", |
| "$$\(patsubst %$$\(ext),%.o,$$\(filter %$$\(ext),$$\(wildcard", Input, "))))\n", |
| "\t$$\(CC) -o $$\@ $$\? $$\(LDFLAGS) $$\(ERL_LDFLAGS) $$\(DRV_LDFLAGS) $$\(LDLIBS) $$\(EXE_LDFLAGS)", |
| case {filename:extension(Output), $(PLATFORM)} of |
| {[], _} -> "\n"; |
| {".so", darwin} -> " -shared\n"; |
| {".dylib", darwin} -> " -shared\n"; |
| {_, darwin} -> "\n"; |
| _ -> " -shared\n" |
| end]) |
| end, |
| [PortSpec(S) || S <- PortSpecs] |
| end, |
| fun() -> |
| case lists:keyfind(plugins, 1, Conf) of |
| false -> ok; |
| {_, Plugins0} -> |
| Plugins = [P || P <- Plugins0, is_tuple(P)], |
| case lists:keyfind('lfe-compile', 1, Plugins) of |
| false -> ok; |
| _ -> Write("\nBUILD_DEPS = lfe lfe.mk\ndep_lfe.mk = git https://github.com/ninenines/lfe.mk master\nDEP_PLUGINS = lfe.mk\n") |
| end |
| end |
| end(), |
| Write("\ninclude $$\(if $$\(ERLANG_MK_FILENAME),$$\(ERLANG_MK_FILENAME),erlang.mk)"), |
| RunPlugin = fun(Plugin, Step) -> |
| case erlang:function_exported(Plugin, Step, 2) of |
| false -> ok; |
| true -> |
| c:cd("$(call core_native_path,$(DEPS_DIR)/$1/)"), |
| Ret = Plugin:Step({config, "", Conf, dict:new(), dict:new(), dict:new(), |
| dict:store(base_dir, "", dict:new())}, undefined), |
| io:format("rebar plugin ~p step ~p ret ~p~n", [Plugin, Step, Ret]) |
| end |
| end, |
| fun() -> |
| case lists:keyfind(plugins, 1, Conf) of |
| false -> ok; |
| {_, Plugins0} -> |
| Plugins = [P || P <- Plugins0, is_atom(P)], |
| [begin |
| case lists:keyfind(deps, 1, Conf) of |
| false -> ok; |
| {_, Deps} -> |
| case lists:keyfind(P, 1, Deps) of |
| false -> ok; |
| _ -> |
| Path = "$(call core_native_path,$(DEPS_DIR)/)" ++ atom_to_list(P), |
| io:format("~s", [os:cmd("$(MAKE) -C $(call core_native_path,$(DEPS_DIR)/$1) " ++ Path)]), |
| io:format("~s", [os:cmd("$(MAKE) -C " ++ Path ++ " IS_DEP=1")]), |
| code:add_patha(Path ++ "/ebin") |
| end |
| end |
| end || P <- Plugins], |
| [case code:load_file(P) of |
| {module, P} -> ok; |
| _ -> |
| case lists:keyfind(plugin_dir, 1, Conf) of |
| false -> ok; |
| {_, PluginsDir} -> |
| ErlFile = "$(call core_native_path,$(DEPS_DIR)/$1/)" ++ PluginsDir ++ "/" ++ atom_to_list(P) ++ ".erl", |
| {ok, P, Bin} = compile:file(ErlFile, [binary]), |
| {module, P} = code:load_binary(P, ErlFile, Bin) |
| end |
| end || P <- Plugins], |
| [RunPlugin(P, preprocess) || P <- Plugins], |
| [RunPlugin(P, pre_compile) || P <- Plugins], |
| [RunPlugin(P, compile) || P <- Plugins] |
| end |
| end(), |
| halt() |
| endef |
| |
| define dep_autopatch_appsrc_script.erl |
| AppSrc = "$(call core_native_path,$(DEPS_DIR)/$1/src/$1.app.src)", |
| AppSrcScript = AppSrc ++ ".script", |
| Conf1 = case file:consult(AppSrc) of |
| {ok, Conf0} -> Conf0; |
| {error, enoent} -> [] |
| end, |
| Bindings0 = erl_eval:new_bindings(), |
| Bindings1 = erl_eval:add_binding('CONFIG', Conf1, Bindings0), |
| Bindings = erl_eval:add_binding('SCRIPT', AppSrcScript, Bindings1), |
| Conf = case file:script(AppSrcScript, Bindings) of |
| {ok, [C]} -> C; |
| {ok, C} -> C |
| end, |
| ok = file:write_file(AppSrc, io_lib:format("~p.~n", [Conf])), |
| halt() |
| endef |
| |
| define dep_autopatch_appsrc.erl |
| AppSrcOut = "$(call core_native_path,$(DEPS_DIR)/$1/src/$1.app.src)", |
| AppSrcIn = case filelib:is_regular(AppSrcOut) of false -> "$(call core_native_path,$(DEPS_DIR)/$1/ebin/$1.app)"; true -> AppSrcOut end, |
| case filelib:is_regular(AppSrcIn) of |
| false -> ok; |
| true -> |
| {ok, [{application, $(1), L0}]} = file:consult(AppSrcIn), |
| L1 = lists:keystore(modules, 1, L0, {modules, []}), |
| L2 = case lists:keyfind(vsn, 1, L1) of |
| {_, git} -> lists:keyreplace(vsn, 1, L1, {vsn, lists:droplast(os:cmd("git -C $(DEPS_DIR)/$1 describe --dirty --tags --always"))}); |
| {_, {cmd, _}} -> lists:keyreplace(vsn, 1, L1, {vsn, "cmd"}); |
| _ -> L1 |
| end, |
| L3 = case lists:keyfind(registered, 1, L2) of false -> [{registered, []}|L2]; _ -> L2 end, |
| ok = file:write_file(AppSrcOut, io_lib:format("~p.~n", [{application, $(1), L3}])), |
| case AppSrcOut of AppSrcIn -> ok; _ -> ok = file:delete(AppSrcIn) end |
| end, |
| halt() |
| endef |
| |
| ifeq ($(CACHE_DEPS),1) |
| |
| define dep_cache_fetch_git |
| mkdir -p $(CACHE_DIR)/git; \ |
| if test -d "$(join $(CACHE_DIR)/git/,$(call dep_name,$1))"; then \ |
| cd $(join $(CACHE_DIR)/git/,$(call dep_name,$1)); \ |
| if ! git checkout -q $(call dep_commit,$1); then \ |
| git remote set-url origin $(call dep_repo,$1) && \ |
| git pull --all && \ |
| git cat-file -e $(call dep_commit,$1) 2>/dev/null; \ |
| fi; \ |
| else \ |
| git clone -q -n -- $(call dep_repo,$1) $(join $(CACHE_DIR)/git/,$(call dep_name,$1)); \ |
| fi; \ |
| git clone -q --branch $(call dep_commit,$1) --single-branch -- $(join $(CACHE_DIR)/git/,$(call dep_name,$1)) $2 |
| endef |
| |
| define dep_fetch_git |
| $(call dep_cache_fetch_git,$1,$(DEPS_DIR)/$(call dep_name,$1)); |
| endef |
| |
| define dep_fetch_git-subfolder |
| mkdir -p $(ERLANG_MK_TMP)/git-subfolder; \ |
| $(call dep_cache_fetch_git,$1,$(ERLANG_MK_TMP)/git-subfolder/$(call dep_name,$1)); \ |
| ln -s $(ERLANG_MK_TMP)/git-subfolder/$(call dep_name,$1)/$(word 4,$(dep_$1)) \ |
| $(DEPS_DIR)/$(call dep_name,$1); |
| endef |
| |
| else |
| |
| define dep_fetch_git |
| git clone -q -n -- $(call dep_repo,$1) $(DEPS_DIR)/$(call dep_name,$1); \ |
| cd $(DEPS_DIR)/$(call dep_name,$1) && git checkout -q $(call dep_commit,$1); |
| endef |
| |
| define dep_fetch_git-subfolder |
| mkdir -p $(ERLANG_MK_TMP)/git-subfolder; \ |
| git clone -q -n -- $(call dep_repo,$1) \ |
| $(ERLANG_MK_TMP)/git-subfolder/$(call dep_name,$1); \ |
| cd $(ERLANG_MK_TMP)/git-subfolder/$(call dep_name,$1) \ |
| && git checkout -q $(call dep_commit,$1); \ |
| ln -s $(ERLANG_MK_TMP)/git-subfolder/$(call dep_name,$1)/$(word 4,$(dep_$1)) \ |
| $(DEPS_DIR)/$(call dep_name,$1); |
| endef |
| |
| endif |
| |
| define dep_fetch_git-submodule |
| git submodule update --init -- $(DEPS_DIR)/$1; |
| endef |
| |
| define dep_fetch_hg |
| hg clone -q -U $(call dep_repo,$(1)) $(DEPS_DIR)/$(call dep_name,$(1)); \ |
| cd $(DEPS_DIR)/$(call dep_name,$(1)) && hg update -q $(call dep_commit,$(1)); |
| endef |
| |
| define dep_fetch_svn |
| svn checkout -q $(call dep_repo,$(1)) $(DEPS_DIR)/$(call dep_name,$(1)); |
| endef |
| |
| define dep_fetch_cp |
| cp -R $(call dep_repo,$(1)) $(DEPS_DIR)/$(call dep_name,$(1)); |
| endef |
| |
| define dep_fetch_ln |
| ln -s $(call dep_repo,$(1)) $(DEPS_DIR)/$(call dep_name,$(1)); |
| endef |
| |
| ifeq ($(CACHE_DEPS),1) |
| |
| # Hex only has a package version. No need to look in the Erlang.mk packages. |
| define dep_fetch_hex |
| mkdir -p $(CACHE_DIR)/hex $(DEPS_DIR)/$1; \ |
| $(eval hex_tar_name=$(if $(word 3,$(dep_$1)),$(word 3,$(dep_$1)),$1)-$(strip $(word 2,$(dep_$1))).tar) \ |
| $(if $(wildcard $(CACHE_DIR)/hex/$(hex_tar_name)),,$(call core_http_get,$(CACHE_DIR)/hex/$(hex_tar_name),\ |
| https://repo.hex.pm/tarballs/$(hex_tar_name);)) \ |
| tar -xOf $(CACHE_DIR)/hex/$(hex_tar_name) contents.tar.gz | tar -C $(DEPS_DIR)/$1 -xzf -; |
| endef |
| |
| else |
| |
| # Hex only has a package version. No need to look in the Erlang.mk packages. |
| define dep_fetch_hex |
| mkdir -p $(ERLANG_MK_TMP)/hex $(DEPS_DIR)/$1; \ |
| $(call core_http_get,$(ERLANG_MK_TMP)/hex/$1.tar,\ |
| https://repo.hex.pm/tarballs/$(if $(word 3,$(dep_$1)),$(word 3,$(dep_$1)),$1)-$(strip $(word 2,$(dep_$1))).tar); \ |
| tar -xOf $(ERLANG_MK_TMP)/hex/$1.tar contents.tar.gz | tar -C $(DEPS_DIR)/$1 -xzf -; |
| endef |
| |
| endif |
| |
| define dep_fetch_fail |
| echo "Error: Unknown or invalid dependency: $(1)." >&2; \ |
| exit 78; |
| endef |
| |
| # Kept for compatibility purposes with older Erlang.mk configuration. |
| define dep_fetch_legacy |
| $(warning WARNING: '$(1)' dependency configuration uses deprecated format.) \ |
| git clone -q -n -- $(word 1,$(dep_$(1))) $(DEPS_DIR)/$(1); \ |
| cd $(DEPS_DIR)/$(1) && git checkout -q $(if $(word 2,$(dep_$(1))),$(word 2,$(dep_$(1))),master); |
| endef |
| |
| define dep_target |
| $(DEPS_DIR)/$(call dep_name,$1): | $(ERLANG_MK_TMP) |
| $(eval DEP_NAME := $(call dep_name,$1)) |
| $(eval DEP_STR := $(if $(filter $1,$(DEP_NAME)),$1,"$1 ($(DEP_NAME))")) |
| $(verbose) if test -d $(APPS_DIR)/$(DEP_NAME); then \ |
| echo "Error: Dependency" $(DEP_STR) "conflicts with application found in $(APPS_DIR)/$(DEP_NAME)." >&2; \ |
| exit 17; \ |
| fi |
| $(verbose) mkdir -p $(DEPS_DIR) |
| $(dep_verbose) $(call dep_fetch_$(strip $(call dep_fetch,$(1))),$(1)) |
| $(verbose) if [ -f $(DEPS_DIR)/$(1)/configure.ac -o -f $(DEPS_DIR)/$(1)/configure.in ] \ |
| && [ ! -f $(DEPS_DIR)/$(1)/configure ]; then \ |
| echo " AUTO " $(DEP_STR); \ |
| cd $(DEPS_DIR)/$(1) && autoreconf -Wall -vif -I m4; \ |
| fi |
| - $(verbose) if [ -f $(DEPS_DIR)/$(DEP_NAME)/configure ]; then \ |
| echo " CONF " $(DEP_STR); \ |
| cd $(DEPS_DIR)/$(DEP_NAME) && ./configure; \ |
| fi |
| ifeq ($(filter $(1),$(NO_AUTOPATCH)),) |
| $(verbose) $$(MAKE) --no-print-directory autopatch-$(DEP_NAME) |
| endif |
| |
| .PHONY: autopatch-$(call dep_name,$1) |
| |
| autopatch-$(call dep_name,$1):: |
| $(verbose) if [ "$1" = "elixir" -a "$(ELIXIR_PATCH)" ]; then \ |
| ln -s lib/elixir/ebin $(DEPS_DIR)/elixir/; \ |
| else \ |
| $$(call dep_autopatch,$(call dep_name,$1)) \ |
| fi |
| endef |
| |
| $(foreach dep,$(BUILD_DEPS) $(DEPS),$(eval $(call dep_target,$(dep)))) |
| |
| ifndef IS_APP |
| clean:: clean-apps |
| |
| clean-apps: |
| $(verbose) set -e; for dep in $(ALL_APPS_DIRS) ; do \ |
| $(MAKE) -C $$dep clean IS_APP=1; \ |
| done |
| |
| distclean:: distclean-apps |
| |
| distclean-apps: |
| $(verbose) set -e; for dep in $(ALL_APPS_DIRS) ; do \ |
| $(MAKE) -C $$dep distclean IS_APP=1; \ |
| done |
| endif |
| |
| ifndef SKIP_DEPS |
| distclean:: distclean-deps |
| |
| distclean-deps: |
| $(gen_verbose) rm -rf $(DEPS_DIR) |
| endif |
| |
| ifeq ($(CACHE_DEPS),1) |
| cacheclean:: cacheclean-git cacheclean-hex |
| |
| cacheclean-git: |
| $(gen_verbose) rm -rf $(CACHE_DIR)/git |
| |
| cacheclean-hex: |
| $(gen_verbose) rm -rf $(CACHE_DIR)/hex |
| endif |
| |
| # Forward-declare variables used in core/deps-tools.mk. This is required |
| # in case plugins use them. |
| |
| ERLANG_MK_RECURSIVE_DEPS_LIST = $(ERLANG_MK_TMP)/recursive-deps-list.log |
| ERLANG_MK_RECURSIVE_DOC_DEPS_LIST = $(ERLANG_MK_TMP)/recursive-doc-deps-list.log |
| ERLANG_MK_RECURSIVE_REL_DEPS_LIST = $(ERLANG_MK_TMP)/recursive-rel-deps-list.log |
| ERLANG_MK_RECURSIVE_TEST_DEPS_LIST = $(ERLANG_MK_TMP)/recursive-test-deps-list.log |
| ERLANG_MK_RECURSIVE_SHELL_DEPS_LIST = $(ERLANG_MK_TMP)/recursive-shell-deps-list.log |
| |
| ERLANG_MK_QUERY_DEPS_FILE = $(ERLANG_MK_TMP)/query-deps.log |
| ERLANG_MK_QUERY_DOC_DEPS_FILE = $(ERLANG_MK_TMP)/query-doc-deps.log |
| ERLANG_MK_QUERY_REL_DEPS_FILE = $(ERLANG_MK_TMP)/query-rel-deps.log |
| ERLANG_MK_QUERY_TEST_DEPS_FILE = $(ERLANG_MK_TMP)/query-test-deps.log |
| ERLANG_MK_QUERY_SHELL_DEPS_FILE = $(ERLANG_MK_TMP)/query-shell-deps.log |
| |
| # Copyright (c) 2024, Loïc Hoguin <essen@ninenines.eu> |
| # This file is part of erlang.mk and subject to the terms of the ISC License. |
| |
| .PHONY: beam-cache-restore-app beam-cache-restore-test clean-beam-cache distclean-beam-cache |
| |
| BEAM_CACHE_DIR ?= $(ERLANG_MK_TMP)/beam-cache |
| PROJECT_BEAM_CACHE_DIR = $(BEAM_CACHE_DIR)/$(PROJECT) |
| |
| clean:: clean-beam-cache |
| |
| clean-beam-cache: |
| $(verbose) rm -rf $(PROJECT_BEAM_CACHE_DIR) |
| |
| distclean:: distclean-beam-cache |
| |
| $(PROJECT_BEAM_CACHE_DIR): |
| $(verbose) mkdir -p $(PROJECT_BEAM_CACHE_DIR) |
| |
| distclean-beam-cache: |
| $(gen_verbose) rm -rf $(BEAM_CACHE_DIR) |
| |
| beam-cache-restore-app: | $(PROJECT_BEAM_CACHE_DIR) |
| $(verbose) rm -rf $(PROJECT_BEAM_CACHE_DIR)/ebin-test |
| ifneq ($(wildcard ebin/),) |
| $(verbose) mv ebin/ $(PROJECT_BEAM_CACHE_DIR)/ebin-test |
| endif |
| ifneq ($(wildcard $(PROJECT_BEAM_CACHE_DIR)/ebin-app),) |
| $(gen_verbose) mv $(PROJECT_BEAM_CACHE_DIR)/ebin-app ebin/ |
| else |
| $(verbose) $(MAKE) --no-print-directory clean-app |
| endif |
| |
| beam-cache-restore-test: | $(PROJECT_BEAM_CACHE_DIR) |
| $(verbose) rm -rf $(PROJECT_BEAM_CACHE_DIR)/ebin-app |
| ifneq ($(wildcard ebin/),) |
| $(verbose) mv ebin/ $(PROJECT_BEAM_CACHE_DIR)/ebin-app |
| endif |
| ifneq ($(wildcard $(PROJECT_BEAM_CACHE_DIR)/ebin-test),) |
| $(gen_verbose) mv $(PROJECT_BEAM_CACHE_DIR)/ebin-test ebin/ |
| else |
| $(verbose) $(MAKE) --no-print-directory clean-app |
| endif |
| |
| # Copyright (c) 2013-2016, Loïc Hoguin <essen@ninenines.eu> |
| # This file is part of erlang.mk and subject to the terms of the ISC License. |
| |
| .PHONY: clean-app |
| |
| # Configuration. |
| |
| ERLC_OPTS ?= -Werror +debug_info +warn_export_vars +warn_shadow_vars \ |
| +warn_obsolete_guard # +bin_opt_info +warn_export_all +warn_missing_spec |
| COMPILE_FIRST ?= |
| COMPILE_FIRST_PATHS = $(addprefix src/,$(addsuffix .erl,$(COMPILE_FIRST))) |
| ERLC_EXCLUDE ?= |
| ERLC_EXCLUDE_PATHS = $(addprefix src/,$(addsuffix .erl,$(ERLC_EXCLUDE))) |
| |
| ERLC_ASN1_OPTS ?= |
| |
| ERLC_MIB_OPTS ?= |
| COMPILE_MIB_FIRST ?= |
| COMPILE_MIB_FIRST_PATHS = $(addprefix mibs/,$(addsuffix .mib,$(COMPILE_MIB_FIRST))) |
| |
| # Verbosity. |
| |
| app_verbose_0 = @echo " APP " $(PROJECT); |
| app_verbose_2 = set -x; |
| app_verbose = $(app_verbose_$(V)) |
| |
| appsrc_verbose_0 = @echo " APP " $(PROJECT).app.src; |
| appsrc_verbose_2 = set -x; |
| appsrc_verbose = $(appsrc_verbose_$(V)) |
| |
| makedep_verbose_0 = @echo " DEPEND" $(PROJECT).d; |
| makedep_verbose_2 = set -x; |
| makedep_verbose = $(makedep_verbose_$(V)) |
| |
| erlc_verbose_0 = @echo " ERLC " $(filter-out $(patsubst %,%.erl,$(ERLC_EXCLUDE)),\ |
| $(filter %.erl %.core,$(?F))); |
| erlc_verbose_2 = set -x; |
| erlc_verbose = $(erlc_verbose_$(V)) |
| |
| xyrl_verbose_0 = @echo " XYRL " $(filter %.xrl %.yrl,$(?F)); |
| xyrl_verbose_2 = set -x; |
| xyrl_verbose = $(xyrl_verbose_$(V)) |
| |
| asn1_verbose_0 = @echo " ASN1 " $(filter %.asn1,$(?F)); |
| asn1_verbose_2 = set -x; |
| asn1_verbose = $(asn1_verbose_$(V)) |
| |
| mib_verbose_0 = @echo " MIB " $(filter %.bin %.mib,$(?F)); |
| mib_verbose_2 = set -x; |
| mib_verbose = $(mib_verbose_$(V)) |
| |
| ifneq ($(wildcard src/),) |
| |
| # Targets. |
| |
| app:: $(if $(wildcard ebin/test),beam-cache-restore-app) deps |
| $(verbose) $(MAKE) --no-print-directory $(PROJECT).d |
| $(verbose) $(MAKE) --no-print-directory app-build |
| |
| ifeq ($(wildcard src/$(PROJECT_MOD).erl),) |
| define app_file |
| {application, '$(PROJECT)', [ |
| {description, "$(PROJECT_DESCRIPTION)"}, |
| {vsn, "$(PROJECT_VERSION)"},$(if $(IS_DEP), |
| {id$(comma)$(space)"$(1)"}$(comma)) |
| {modules, [$(call comma_list,$(2))]}, |
| {registered, []}, |
| {applications, [$(call comma_list,kernel stdlib $(OTP_DEPS) $(LOCAL_DEPS) $(OPTIONAL_DEPS) $(foreach dep,$(DEPS),$(call dep_name,$(dep))))]}, |
| {optional_applications, [$(call comma_list,$(OPTIONAL_DEPS))]}, |
| {env, $(subst \,\\,$(PROJECT_ENV))}$(if $(findstring {,$(PROJECT_APP_EXTRA_KEYS)),$(comma)$(newline)$(tab)$(subst \,\\,$(PROJECT_APP_EXTRA_KEYS)),) |
| ]}. |
| endef |
| else |
| define app_file |
| {application, '$(PROJECT)', [ |
| {description, "$(PROJECT_DESCRIPTION)"}, |
| {vsn, "$(PROJECT_VERSION)"},$(if $(IS_DEP), |
| {id$(comma)$(space)"$(1)"}$(comma)) |
| {modules, [$(call comma_list,$(2))]}, |
| {registered, [$(call comma_list,$(PROJECT)_sup $(PROJECT_REGISTERED))]}, |
| {applications, [$(call comma_list,kernel stdlib $(OTP_DEPS) $(LOCAL_DEPS) $(OPTIONAL_DEPS) $(foreach dep,$(DEPS),$(call dep_name,$(dep))))]}, |
| {optional_applications, [$(call comma_list,$(OPTIONAL_DEPS))]}, |
| {mod, {$(PROJECT_MOD), []}}, |
| {env, $(subst \,\\,$(PROJECT_ENV))}$(if $(findstring {,$(PROJECT_APP_EXTRA_KEYS)),$(comma)$(newline)$(tab)$(subst \,\\,$(PROJECT_APP_EXTRA_KEYS)),) |
| ]}. |
| endef |
| endif |
| |
| app-build: ebin/$(PROJECT).app |
| $(verbose) : |
| |
| # Source files. |
| |
| ALL_SRC_FILES := $(sort $(call core_find,src/,*)) |
| |
| ERL_FILES := $(filter %.erl,$(ALL_SRC_FILES)) |
| CORE_FILES := $(filter %.core,$(ALL_SRC_FILES)) |
| |
| # ASN.1 files. |
| |
| ifneq ($(wildcard asn1/),) |
| ASN1_FILES = $(sort $(call core_find,asn1/,*.asn1)) |
| ERL_FILES += $(addprefix src/,$(patsubst %.asn1,%.erl,$(notdir $(ASN1_FILES)))) |
| |
| define compile_asn1 |
| $(verbose) mkdir -p include/ |
| $(asn1_verbose) erlc -v -I include/ -o asn1/ +noobj $(ERLC_ASN1_OPTS) $(1) |
| $(verbose) mv asn1/*.erl src/ |
| -$(verbose) mv asn1/*.hrl include/ |
| $(verbose) mv asn1/*.asn1db include/ |
| endef |
| |
| $(PROJECT).d:: $(ASN1_FILES) |
| $(if $(strip $?),$(call compile_asn1,$?)) |
| endif |
| |
| # SNMP MIB files. |
| |
| ifneq ($(wildcard mibs/),) |
| MIB_FILES = $(sort $(call core_find,mibs/,*.mib)) |
| |
| $(PROJECT).d:: $(COMPILE_MIB_FIRST_PATHS) $(MIB_FILES) |
| $(verbose) mkdir -p include/ priv/mibs/ |
| $(mib_verbose) erlc -v $(ERLC_MIB_OPTS) -o priv/mibs/ -I priv/mibs/ $? |
| $(mib_verbose) erlc -o include/ -- $(addprefix priv/mibs/,$(patsubst %.mib,%.bin,$(notdir $?))) |
| endif |
| |
| # Leex and Yecc files. |
| |
| XRL_FILES := $(filter %.xrl,$(ALL_SRC_FILES)) |
| XRL_ERL_FILES = $(addprefix src/,$(patsubst %.xrl,%.erl,$(notdir $(XRL_FILES)))) |
| ERL_FILES += $(XRL_ERL_FILES) |
| |
| YRL_FILES := $(filter %.yrl,$(ALL_SRC_FILES)) |
| YRL_ERL_FILES = $(addprefix src/,$(patsubst %.yrl,%.erl,$(notdir $(YRL_FILES)))) |
| ERL_FILES += $(YRL_ERL_FILES) |
| |
| $(PROJECT).d:: $(XRL_FILES) $(YRL_FILES) |
| $(if $(strip $?),$(xyrl_verbose) erlc -v -o src/ $(YRL_ERLC_OPTS) $?) |
| |
| # Erlang and Core Erlang files. |
| |
| define makedep.erl |
| E = ets:new(makedep, [bag]), |
| G = digraph:new([acyclic]), |
| ErlFiles = lists:usort(string:tokens("$(ERL_FILES)", " ")), |
| DepsDir = "$(call core_native_path,$(DEPS_DIR))", |
| AppsDir = "$(call core_native_path,$(APPS_DIR))", |
| DepsDirsSrc = "$(if $(wildcard $(DEPS_DIR)/*/src), $(call core_native_path,$(wildcard $(DEPS_DIR)/*/src)))", |
| DepsDirsInc = "$(if $(wildcard $(DEPS_DIR)/*/include), $(call core_native_path,$(wildcard $(DEPS_DIR)/*/include)))", |
| AppsDirsSrc = "$(if $(wildcard $(APPS_DIR)/*/src), $(call core_native_path,$(wildcard $(APPS_DIR)/*/src)))", |
| AppsDirsInc = "$(if $(wildcard $(APPS_DIR)/*/include), $(call core_native_path,$(wildcard $(APPS_DIR)/*/include)))", |
| DepsDirs = lists:usort(string:tokens(DepsDirsSrc++DepsDirsInc, " ")), |
| AppsDirs = lists:usort(string:tokens(AppsDirsSrc++AppsDirsInc, " ")), |
| Modules = [{list_to_atom(filename:basename(F, ".erl")), F} || F <- ErlFiles], |
| Add = fun (Mod, Dep) -> |
| case lists:keyfind(Dep, 1, Modules) of |
| false -> ok; |
| {_, DepFile} -> |
| {_, ModFile} = lists:keyfind(Mod, 1, Modules), |
| ets:insert(E, {ModFile, DepFile}), |
| digraph:add_vertex(G, Mod), |
| digraph:add_vertex(G, Dep), |
| digraph:add_edge(G, Mod, Dep) |
| end |
| end, |
| AddHd = fun (F, Mod, DepFile) -> |
| case file:open(DepFile, [read]) of |
| {error, enoent} -> |
| ok; |
| {ok, Fd} -> |
| {_, ModFile} = lists:keyfind(Mod, 1, Modules), |
| case ets:match(E, {ModFile, DepFile}) of |
| [] -> |
| ets:insert(E, {ModFile, DepFile}), |
| F(F, Fd, Mod,0); |
| _ -> ok |
| end |
| end |
| end, |
| SearchHrl = fun |
| F(_Hrl, []) -> {error,enoent}; |
| F(Hrl, [Dir|Dirs]) -> |
| HrlF = filename:join([Dir,Hrl]), |
| case filelib:is_file(HrlF) of |
| true -> |
| {ok, HrlF}; |
| false -> F(Hrl,Dirs) |
| end |
| end, |
| Attr = fun |
| (_F, Mod, behavior, Dep) -> |
| Add(Mod, Dep); |
| (_F, Mod, behaviour, Dep) -> |
| Add(Mod, Dep); |
| (_F, Mod, compile, {parse_transform, Dep}) -> |
| Add(Mod, Dep); |
| (_F, Mod, compile, Opts) when is_list(Opts) -> |
| case proplists:get_value(parse_transform, Opts) of |
| undefined -> ok; |
| Dep -> Add(Mod, Dep) |
| end; |
| (F, Mod, include, Hrl) -> |
| case SearchHrl(Hrl, ["src", "include",AppsDir,DepsDir]++AppsDirs++DepsDirs) of |
| {ok, FoundHrl} -> AddHd(F, Mod, FoundHrl); |
| {error, _} -> false |
| end; |
| (F, Mod, include_lib, Hrl) -> |
| case SearchHrl(Hrl, ["src", "include",AppsDir,DepsDir]++AppsDirs++DepsDirs) of |
| {ok, FoundHrl} -> AddHd(F, Mod, FoundHrl); |
| {error, _} -> false |
| end; |
| (F, Mod, import, {Imp, _}) -> |
| IsFile = |
| case lists:keyfind(Imp, 1, Modules) of |
| false -> false; |
| {_, FilePath} -> filelib:is_file(FilePath) |
| end, |
| case IsFile of |
| false -> ok; |
| true -> Add(Mod, Imp) |
| end; |
| (_, _, _, _) -> ok |
| end, |
| MakeDepend = fun |
| (F, Fd, Mod, StartLocation) -> |
| case io:parse_erl_form(Fd, undefined, StartLocation) of |
| {ok, AbsData, EndLocation} -> |
| case AbsData of |
| {attribute, _, Key, Value} -> |
| Attr(F, Mod, Key, Value), |
| F(F, Fd, Mod, EndLocation); |
| _ -> F(F, Fd, Mod, EndLocation) |
| end; |
| {eof, _ } -> file:close(Fd); |
| {error, ErrorDescription } -> |
| file:close(Fd); |
| {error, ErrorInfo, ErrorLocation} -> |
| F(F, Fd, Mod, ErrorLocation) |
| end, |
| ok |
| end, |
| [begin |
| Mod = list_to_atom(filename:basename(F, ".erl")), |
| case file:open(F, [read]) of |
| {ok, Fd} -> MakeDepend(MakeDepend, Fd, Mod,0); |
| {error, enoent} -> ok |
| end |
| end || F <- ErlFiles], |
| Depend = sofs:to_external(sofs:relation_to_family(sofs:relation(ets:tab2list(E)))), |
| CompileFirst = [X || X <- lists:reverse(digraph_utils:topsort(G)), [] =/= digraph:in_neighbours(G, X)], |
| TargetPath = fun(Target) -> |
| case lists:keyfind(Target, 1, Modules) of |
| false -> ""; |
| {_, DepFile} -> |
| DirSubname = tl(string:tokens(filename:dirname(DepFile), "/")), |
| string:join(DirSubname ++ [atom_to_list(Target)], "/") |
| end |
| end, |
| Output0 = [ |
| "# Generated by Erlang.mk. Edit at your own risk!\n\n", |
| [[F, "::", [[" ", D] || D <- Deps], "; @touch \$$@\n"] || {F, Deps} <- Depend], |
| "\nCOMPILE_FIRST +=", [[" ", TargetPath(CF)] || CF <- CompileFirst], "\n" |
| ], |
| Output = case "é" of |
| [233] -> unicode:characters_to_binary(Output0); |
| _ -> Output0 |
| end, |
| ok = file:write_file("$(1)", Output), |
| halt() |
| endef |
| |
| ifeq ($(if $(NO_MAKEDEP),$(wildcard $(PROJECT).d),),) |
| $(PROJECT).d:: $(ERL_FILES) $(call core_find,include/,*.hrl) $(MAKEFILE_LIST) |
| $(makedep_verbose) $(call erlang,$(call makedep.erl,$@)) |
| endif |
| |
| ifeq ($(IS_APP)$(IS_DEP),) |
| ifneq ($(words $(ERL_FILES) $(CORE_FILES) $(ASN1_FILES) $(MIB_FILES) $(XRL_FILES) $(YRL_FILES)),0) |
| # Rebuild everything when the Makefile changes. |
| $(ERLANG_MK_TMP)/last-makefile-change: $(MAKEFILE_LIST) | $(ERLANG_MK_TMP) |
| $(verbose) if test -f $@; then \ |
| touch $(ERL_FILES) $(CORE_FILES) $(ASN1_FILES) $(MIB_FILES) $(XRL_FILES) $(YRL_FILES); \ |
| touch -c $(PROJECT).d; \ |
| fi |
| $(verbose) touch $@ |
| |
| $(ERL_FILES) $(CORE_FILES) $(ASN1_FILES) $(MIB_FILES) $(XRL_FILES) $(YRL_FILES):: $(ERLANG_MK_TMP)/last-makefile-change |
| ebin/$(PROJECT).app:: $(ERLANG_MK_TMP)/last-makefile-change |
| endif |
| endif |
| |
| $(PROJECT).d:: |
| $(verbose) : |
| |
| include $(wildcard $(PROJECT).d) |
| |
| ebin/$(PROJECT).app:: ebin/ |
| |
| ebin/: |
| $(verbose) mkdir -p ebin/ |
| |
| define compile_erl |
| $(erlc_verbose) erlc -v $(if $(IS_DEP),$(filter-out -Werror,$(ERLC_OPTS)),$(ERLC_OPTS)) -o ebin/ \ |
| -pa ebin/ -I include/ $(filter-out $(ERLC_EXCLUDE_PATHS),$(COMPILE_FIRST_PATHS) $(1)) |
| endef |
| |
| define validate_app_file |
| case file:consult("ebin/$(PROJECT).app") of |
| {ok, _} -> halt(); |
| _ -> halt(1) |
| end |
| endef |
| |
| ebin/$(PROJECT).app:: $(ERL_FILES) $(CORE_FILES) $(wildcard src/$(PROJECT).app.src) |
| $(eval FILES_TO_COMPILE := $(filter-out src/$(PROJECT).app.src,$?)) |
| $(if $(strip $(FILES_TO_COMPILE)),$(call compile_erl,$(FILES_TO_COMPILE))) |
| # Older git versions do not have the --first-parent flag. Do without in that case. |
| $(eval GITDESCRIBE := $(shell git describe --dirty --abbrev=7 --tags --always --first-parent 2>/dev/null \ |
| || git describe --dirty --abbrev=7 --tags --always 2>/dev/null || true)) |
| $(eval MODULES := $(patsubst %,'%',$(sort $(notdir $(basename \ |
| $(filter-out $(ERLC_EXCLUDE_PATHS),$(ERL_FILES) $(CORE_FILES) $(BEAM_FILES))))))) |
| ifeq ($(wildcard src/$(PROJECT).app.src),) |
| $(app_verbose) printf '$(subst %,%%,$(subst $(newline),\n,$(subst ','\'',$(call app_file,$(GITDESCRIBE),$(MODULES)))))' \ |
| > ebin/$(PROJECT).app |
| $(verbose) if ! $(call erlang,$(call validate_app_file)); then \ |
| echo "The .app file produced is invalid. Please verify the value of PROJECT_ENV." >&2; \ |
| exit 1; \ |
| fi |
| else |
| $(verbose) if [ -z "$$(grep -e '^[^%]*{\s*modules\s*,' src/$(PROJECT).app.src)" ]; then \ |
| echo "Empty modules entry not found in $(PROJECT).app.src. Please consult the erlang.mk documentation for instructions." >&2; \ |
| exit 1; \ |
| fi |
| $(appsrc_verbose) cat src/$(PROJECT).app.src \ |
| | sed "s/{[[:space:]]*modules[[:space:]]*,[[:space:]]*\[\]}/{modules, \[$(call comma_list,$(MODULES))\]}/" \ |
| | sed "s/{id,[[:space:]]*\"git\"}/{id, \"$(subst /,\/,$(GITDESCRIBE))\"}/" \ |
| > ebin/$(PROJECT).app |
| endif |
| ifneq ($(wildcard src/$(PROJECT).appup),) |
| $(verbose) cp src/$(PROJECT).appup ebin/ |
| endif |
| |
| clean:: clean-app |
| |
| clean-app: |
| $(gen_verbose) rm -rf $(PROJECT).d ebin/ priv/mibs/ $(XRL_ERL_FILES) $(YRL_ERL_FILES) \ |
| $(addprefix include/,$(patsubst %.mib,%.hrl,$(notdir $(MIB_FILES)))) \ |
| $(addprefix include/,$(patsubst %.asn1,%.hrl,$(notdir $(ASN1_FILES)))) \ |
| $(addprefix include/,$(patsubst %.asn1,%.asn1db,$(notdir $(ASN1_FILES)))) \ |
| $(addprefix src/,$(patsubst %.asn1,%.erl,$(notdir $(ASN1_FILES)))) |
| |
| endif |
| |
| # Copyright (c) 2016, Loïc Hoguin <essen@ninenines.eu> |
| # Copyright (c) 2015, Viktor Söderqvist <viktor@zuiderkwast.se> |
| # This file is part of erlang.mk and subject to the terms of the ISC License. |
| |
| .PHONY: docs-deps |
| |
| # Configuration. |
| |
| ALL_DOC_DEPS_DIRS = $(addprefix $(DEPS_DIR)/,$(DOC_DEPS)) |
| |
| # Targets. |
| |
| $(foreach dep,$(DOC_DEPS),$(eval $(call dep_target,$(dep)))) |
| |
| ifneq ($(SKIP_DEPS),) |
| doc-deps: |
| else |
| doc-deps: $(ALL_DOC_DEPS_DIRS) |
| $(verbose) set -e; for dep in $(ALL_DOC_DEPS_DIRS) ; do $(MAKE) -C $$dep IS_DEP=1; done |
| endif |
| |
| # Copyright (c) 2015-2016, Loïc Hoguin <essen@ninenines.eu> |
| # This file is part of erlang.mk and subject to the terms of the ISC License. |
| |
| .PHONY: rel-deps |
| |
| # Configuration. |
| |
| ALL_REL_DEPS_DIRS = $(addprefix $(DEPS_DIR)/,$(REL_DEPS)) |
| |
| # Targets. |
| |
| $(foreach dep,$(REL_DEPS),$(eval $(call dep_target,$(dep)))) |
| |
| ifneq ($(SKIP_DEPS),) |
| rel-deps: |
| else |
| rel-deps: $(ALL_REL_DEPS_DIRS) |
| $(verbose) set -e; for dep in $(ALL_REL_DEPS_DIRS) ; do $(MAKE) -C $$dep; done |
| endif |
| |
| # Copyright (c) 2015-2016, Loïc Hoguin <essen@ninenines.eu> |
| # This file is part of erlang.mk and subject to the terms of the ISC License. |
| |
| .PHONY: test-deps test-dir test-build clean-test-dir |
| |
| # Configuration. |
| |
| TEST_DIR ?= $(CURDIR)/test |
| |
| ALL_TEST_DEPS_DIRS = $(addprefix $(DEPS_DIR)/,$(TEST_DEPS)) |
| |
| TEST_ERLC_OPTS ?= +debug_info +warn_export_vars +warn_shadow_vars +warn_obsolete_guard |
| TEST_ERLC_OPTS += -DTEST=1 |
| |
| # Targets. |
| |
| $(foreach dep,$(TEST_DEPS),$(eval $(call dep_target,$(dep)))) |
| |
| ifneq ($(SKIP_DEPS),) |
| test-deps: |
| else |
| test-deps: $(ALL_TEST_DEPS_DIRS) |
| $(verbose) set -e; for dep in $(ALL_TEST_DEPS_DIRS) ; do \ |
| if [ -z "$(strip $(FULL))" ] && [ ! -L $$dep ] && [ -f $$dep/ebin/dep_built ]; then \ |
| :; \ |
| else \ |
| $(MAKE) -C $$dep IS_DEP=1; \ |
| if [ ! -L $$dep ] && [ -d $$dep/ebin ]; then touch $$dep/ebin/dep_built; fi; \ |
| fi \ |
| done |
| endif |
| |
| ifneq ($(wildcard $(TEST_DIR)),) |
| test-dir: $(ERLANG_MK_TMP)/$(PROJECT).last-testdir-build |
| @: |
| |
| test_erlc_verbose_0 = @echo " ERLC " $(filter-out $(patsubst %,%.erl,$(ERLC_EXCLUDE)),\ |
| $(filter %.erl %.core,$(notdir $(FILES_TO_COMPILE)))); |
| test_erlc_verbose_2 = set -x; |
| test_erlc_verbose = $(test_erlc_verbose_$(V)) |
| |
| define compile_test_erl |
| $(test_erlc_verbose) erlc -v $(TEST_ERLC_OPTS) -o $(TEST_DIR) \ |
| -pa ebin/ -I include/ $(1) |
| endef |
| |
| ERL_TEST_FILES = $(call core_find,$(TEST_DIR)/,*.erl) |
| |
| $(ERLANG_MK_TMP)/$(PROJECT).last-testdir-build: $(ERL_TEST_FILES) $(MAKEFILE_LIST) |
| # When we have to recompile files in src/ the .d file always gets rebuilt. |
| # Therefore we want to ignore it when rebuilding test files. |
| $(eval FILES_TO_COMPILE := $(if $(filter $(filter-out $(PROJECT).d,$(MAKEFILE_LIST)),$?),$(filter $(ERL_TEST_FILES),$^),$(filter $(ERL_TEST_FILES),$?))) |
| $(if $(strip $(FILES_TO_COMPILE)),$(call compile_test_erl,$(FILES_TO_COMPILE)) && touch $@) |
| endif |
| |
| test-build:: IS_TEST=1 |
| test-build:: ERLC_OPTS=$(TEST_ERLC_OPTS) |
| test-build:: $(if $(wildcard src),$(if $(wildcard ebin/test),,beam-cache-restore-test)) $(if $(IS_APP),,deps test-deps) |
| # We already compiled everything when IS_APP=1. |
| ifndef IS_APP |
| ifneq ($(wildcard src),) |
| $(verbose) $(MAKE) --no-print-directory $(PROJECT).d ERLC_OPTS="$(call escape_dquotes,$(TEST_ERLC_OPTS))" |
| $(verbose) $(MAKE) --no-print-directory app-build ERLC_OPTS="$(call escape_dquotes,$(TEST_ERLC_OPTS))" |
| $(gen_verbose) touch ebin/test |
| endif |
| ifneq ($(wildcard $(TEST_DIR)),) |
| $(verbose) $(MAKE) --no-print-directory test-dir ERLC_OPTS="$(call escape_dquotes,$(TEST_ERLC_OPTS))" |
| endif |
| endif |
| |
| # Roughly the same as test-build, but when IS_APP=1. |
| # We only care about compiling the current application. |
| ifdef IS_APP |
| test-build-app:: ERLC_OPTS=$(TEST_ERLC_OPTS) |
| test-build-app:: deps test-deps |
| ifneq ($(wildcard src),) |
| $(verbose) $(MAKE) --no-print-directory $(PROJECT).d ERLC_OPTS="$(call escape_dquotes,$(TEST_ERLC_OPTS))" |
| $(verbose) $(MAKE) --no-print-directory app-build ERLC_OPTS="$(call escape_dquotes,$(TEST_ERLC_OPTS))" |
| $(gen_verbose) touch ebin/test |
| endif |
| ifneq ($(wildcard $(TEST_DIR)),) |
| $(verbose) $(MAKE) --no-print-directory test-dir ERLC_OPTS="$(call escape_dquotes,$(TEST_ERLC_OPTS))" |
| endif |
| endif |
| |
| clean:: clean-test-dir |
| |
| clean-test-dir: |
| ifneq ($(wildcard $(TEST_DIR)/*.beam),) |
| $(gen_verbose) rm -f $(TEST_DIR)/*.beam $(ERLANG_MK_TMP)/$(PROJECT).last-testdir-build |
| endif |
| |
| # Copyright (c) 2015-2016, Loïc Hoguin <essen@ninenines.eu> |
| # This file is part of erlang.mk and subject to the terms of the ISC License. |
| |
| .PHONY: rebar.config |
| |
| # We strip out -Werror because we don't want to fail due to |
| # warnings when used as a dependency. |
| |
| compat_prepare_erlc_opts = $(shell echo "$1" | sed 's/, */,/g') |
| |
| define compat_convert_erlc_opts |
| $(if $(filter-out -Werror,$1),\ |
| $(if $(findstring +,$1),\ |
| $(shell echo $1 | cut -b 2-))) |
| endef |
| |
| define compat_erlc_opts_to_list |
| [$(call comma_list,$(foreach o,$(call compat_prepare_erlc_opts,$1),$(call compat_convert_erlc_opts,$o)))] |
| endef |
| |
| define compat_rebar_config |
| {deps, [ |
| $(call comma_list,$(foreach d,$(DEPS),\ |
| $(if $(filter hex,$(call dep_fetch,$d)),\ |
| {$(call dep_name,$d)$(comma)"$(call dep_repo,$d)"},\ |
| {$(call dep_name,$d)$(comma)".*"$(comma){git,"$(call dep_repo,$d)"$(comma)"$(call dep_commit,$d)"}}))) |
| ]}. |
| {erl_opts, $(call compat_erlc_opts_to_list,$(ERLC_OPTS))}. |
| endef |
| |
| rebar.config: |
| $(gen_verbose) $(call core_render,compat_rebar_config,rebar.config) |
| |
| # Copyright (c) 2015-2016, Loïc Hoguin <essen@ninenines.eu> |
| # This file is part of erlang.mk and subject to the terms of the ISC License. |
| |
| ifeq ($(filter asciideck,$(DEPS) $(DOC_DEPS)),asciideck) |
| |
| .PHONY: asciidoc asciidoc-guide asciidoc-manual install-asciidoc distclean-asciidoc-guide distclean-asciidoc-manual |
| |
| # Core targets. |
| |
| docs:: asciidoc |
| |
| distclean:: distclean-asciidoc-guide distclean-asciidoc-manual |
| |
| # Plugin-specific targets. |
| |
| asciidoc: asciidoc-guide asciidoc-manual |
| |
| # User guide. |
| |
| ifeq ($(wildcard doc/src/guide/book.asciidoc),) |
| asciidoc-guide: |
| else |
| asciidoc-guide: distclean-asciidoc-guide doc-deps |
| a2x -v -f pdf doc/src/guide/book.asciidoc && mv doc/src/guide/book.pdf doc/guide.pdf |
| a2x -v -f chunked doc/src/guide/book.asciidoc && mv doc/src/guide/book.chunked/ doc/html/ |
| |
| distclean-asciidoc-guide: |
| $(gen_verbose) rm -rf doc/html/ doc/guide.pdf |
| endif |
| |
| # Man pages. |
| |
| ASCIIDOC_MANUAL_FILES := $(wildcard doc/src/manual/*.asciidoc) |
| |
| ifeq ($(ASCIIDOC_MANUAL_FILES),) |
| asciidoc-manual: |
| else |
| |
| # Configuration. |
| |
| MAN_INSTALL_PATH ?= /usr/local/share/man |
| MAN_SECTIONS ?= 3 7 |
| MAN_PROJECT ?= $(shell echo $(PROJECT) | sed 's/^./\U&\E/') |
| MAN_VERSION ?= $(PROJECT_VERSION) |
| |
| # Plugin-specific targets. |
| |
| define asciidoc2man.erl |
| try |
| [begin |
| io:format(" ADOC ~s~n", [F]), |
| ok = asciideck:to_manpage(asciideck:parse_file(F), #{ |
| compress => gzip, |
| outdir => filename:dirname(F), |
| extra2 => "$(MAN_PROJECT) $(MAN_VERSION)", |
| extra3 => "$(MAN_PROJECT) Function Reference" |
| }) |
| end || F <- [$(shell echo $(addprefix $(comma)\",$(addsuffix \",$1)) | sed 's/^.//')]], |
| halt(0) |
| catch C:E$(if $V,:S) -> |
| io:format("Exception: ~p:~p~n$(if $V,Stacktrace: ~p~n)", [C, E$(if $V,$(comma) S)]), |
| halt(1) |
| end. |
| endef |
| |
| asciidoc-manual:: doc-deps |
| |
| asciidoc-manual:: $(ASCIIDOC_MANUAL_FILES) |
| $(gen_verbose) $(call erlang,$(call asciidoc2man.erl,$?)) |
| $(verbose) $(foreach s,$(MAN_SECTIONS),mkdir -p doc/man$s/ && mv doc/src/manual/*.$s.gz doc/man$s/;) |
| |
| install-docs:: install-asciidoc |
| |
| install-asciidoc: asciidoc-manual |
| $(foreach s,$(MAN_SECTIONS),\ |
| mkdir -p $(MAN_INSTALL_PATH)/man$s/ && \ |
| install -g `id -g` -o `id -u` -m 0644 doc/man$s/*.gz $(MAN_INSTALL_PATH)/man$s/;) |
| |
| distclean-asciidoc-manual: |
| $(gen_verbose) rm -rf $(addprefix doc/man,$(MAN_SECTIONS)) |
| endif |
| endif |
| |
| # Copyright (c) 2014-2016, Loïc Hoguin <essen@ninenines.eu> |
| # This file is part of erlang.mk and subject to the terms of the ISC License. |
| |
| .PHONY: bootstrap bootstrap-lib bootstrap-rel new list-templates |
| |
| # Core targets. |
| |
| help:: |
| $(verbose) printf "%s\n" "" \ |
| "Bootstrap targets:" \ |
| " bootstrap Generate a skeleton of an OTP application" \ |
| " bootstrap-lib Generate a skeleton of an OTP library" \ |
| " bootstrap-rel Generate the files needed to build a release" \ |
| " new-app in=NAME Create a new local OTP application NAME" \ |
| " new-lib in=NAME Create a new local OTP library NAME" \ |
| " new t=TPL n=NAME Generate a module NAME based on the template TPL" \ |
| " new t=T n=N in=APP Generate a module NAME based on the template TPL in APP" \ |
| " list-templates List available templates" |
| |
| # Bootstrap templates. |
| |
| define bs_appsrc |
| {application, $p, [ |
| {description, ""}, |
| {vsn, "0.1.0"}, |
| {id, "git"}, |
| {modules, []}, |
| {registered, []}, |
| {applications, [ |
| kernel, |
| stdlib |
| ]}, |
| {mod, {$p_app, []}}, |
| {env, []} |
| ]}. |
| endef |
| |
| define bs_appsrc_lib |
| {application, $p, [ |
| {description, ""}, |
| {vsn, "0.1.0"}, |
| {id, "git"}, |
| {modules, []}, |
| {registered, []}, |
| {applications, [ |
| kernel, |
| stdlib |
| ]} |
| ]}. |
| endef |
| |
| # To prevent autocompletion issues with ZSH, we add "include erlang.mk" |
| # separately during the actual bootstrap. |
| define bs_Makefile |
| PROJECT = $p |
| PROJECT_DESCRIPTION = New project |
| PROJECT_VERSION = 0.1.0 |
| $(if $(SP), |
| # Whitespace to be used when creating files from templates. |
| SP = $(SP) |
| ) |
| endef |
| |
| define bs_apps_Makefile |
| PROJECT = $p |
| PROJECT_DESCRIPTION = New project |
| PROJECT_VERSION = 0.1.0 |
| $(if $(SP), |
| # Whitespace to be used when creating files from templates. |
| SP = $(SP) |
| ) |
| # Make sure we know where the applications are located. |
| ROOT_DIR ?= $(call core_relpath,$(dir $(ERLANG_MK_FILENAME)),$(APPS_DIR)/app) |
| APPS_DIR ?= .. |
| DEPS_DIR ?= $(call core_relpath,$(DEPS_DIR),$(APPS_DIR)/app) |
| |
| include $$(ROOT_DIR)/erlang.mk |
| endef |
| |
| define bs_app |
| -module($p_app). |
| -behaviour(application). |
| |
| -export([start/2]). |
| -export([stop/1]). |
| |
| start(_Type, _Args) -> |
| $p_sup:start_link(). |
| |
| stop(_State) -> |
| ok. |
| endef |
| |
| define bs_relx_config |
| {release, {$p_release, "1"}, [$p, sasl, runtime_tools]}. |
| {dev_mode, false}. |
| {include_erts, true}. |
| {extended_start_script, true}. |
| {sys_config, "config/sys.config"}. |
| {vm_args, "config/vm.args"}. |
| endef |
| |
| define bs_sys_config |
| [ |
| ]. |
| endef |
| |
| define bs_vm_args |
| -name $p@127.0.0.1 |
| -setcookie $p |
| -heart |
| endef |
| |
| # Normal templates. |
| |
| define tpl_supervisor |
| -module($(n)). |
| -behaviour(supervisor). |
| |
| -export([start_link/0]). |
| -export([init/1]). |
| |
| start_link() -> |
| supervisor:start_link({local, ?MODULE}, ?MODULE, []). |
| |
| init([]) -> |
| Procs = [], |
| {ok, {{one_for_one, 1, 5}, Procs}}. |
| endef |
| |
| define tpl_gen_server |
| -module($(n)). |
| -behaviour(gen_server). |
| |
| %% API. |
| -export([start_link/0]). |
| |
| %% gen_server. |
| -export([init/1]). |
| -export([handle_call/3]). |
| -export([handle_cast/2]). |
| -export([handle_info/2]). |
| -export([terminate/2]). |
| -export([code_change/3]). |
| |
| -record(state, { |
| }). |
| |
| %% API. |
| |
| -spec start_link() -> {ok, pid()}. |
| start_link() -> |
| gen_server:start_link(?MODULE, [], []). |
| |
| %% gen_server. |
| |
| init([]) -> |
| {ok, #state{}}. |
| |
| handle_call(_Request, _From, State) -> |
| {reply, ignored, State}. |
| |
| handle_cast(_Msg, State) -> |
| {noreply, State}. |
| |
| handle_info(_Info, State) -> |
| {noreply, State}. |
| |
| terminate(_Reason, _State) -> |
| ok. |
| |
| code_change(_OldVsn, State, _Extra) -> |
| {ok, State}. |
| endef |
| |
| define tpl_module |
| -module($(n)). |
| -export([]). |
| endef |
| |
| define tpl_cowboy_http |
| -module($(n)). |
| -behaviour(cowboy_http_handler). |
| |
| -export([init/3]). |
| -export([handle/2]). |
| -export([terminate/3]). |
| |
| -record(state, { |
| }). |
| |
| init(_, Req, _Opts) -> |
| {ok, Req, #state{}}. |
| |
| handle(Req, State=#state{}) -> |
| {ok, Req2} = cowboy_req:reply(200, Req), |
| {ok, Req2, State}. |
| |
| terminate(_Reason, _Req, _State) -> |
| ok. |
| endef |
| |
| define tpl_gen_fsm |
| -module($(n)). |
| -behaviour(gen_fsm). |
| |
| %% API. |
| -export([start_link/0]). |
| |
| %% gen_fsm. |
| -export([init/1]). |
| -export([state_name/2]). |
| -export([handle_event/3]). |
| -export([state_name/3]). |
| -export([handle_sync_event/4]). |
| -export([handle_info/3]). |
| -export([terminate/3]). |
| -export([code_change/4]). |
| |
| -record(state, { |
| }). |
| |
| %% API. |
| |
| -spec start_link() -> {ok, pid()}. |
| start_link() -> |
| gen_fsm:start_link(?MODULE, [], []). |
| |
| %% gen_fsm. |
| |
| init([]) -> |
| {ok, state_name, #state{}}. |
| |
| state_name(_Event, StateData) -> |
| {next_state, state_name, StateData}. |
| |
| handle_event(_Event, StateName, StateData) -> |
| {next_state, StateName, StateData}. |
| |
| state_name(_Event, _From, StateData) -> |
| {reply, ignored, state_name, StateData}. |
| |
| handle_sync_event(_Event, _From, StateName, StateData) -> |
| {reply, ignored, StateName, StateData}. |
| |
| handle_info(_Info, StateName, StateData) -> |
| {next_state, StateName, StateData}. |
| |
| terminate(_Reason, _StateName, _StateData) -> |
| ok. |
| |
| code_change(_OldVsn, StateName, StateData, _Extra) -> |
| {ok, StateName, StateData}. |
| endef |
| |
| define tpl_gen_statem |
| -module($(n)). |
| -behaviour(gen_statem). |
| |
| %% API. |
| -export([start_link/0]). |
| |
| %% gen_statem. |
| -export([callback_mode/0]). |
| -export([init/1]). |
| -export([state_name/3]). |
| -export([handle_event/4]). |
| -export([terminate/3]). |
| -export([code_change/4]). |
| |
| -record(state, { |
| }). |
| |
| %% API. |
| |
| -spec start_link() -> {ok, pid()}. |
| start_link() -> |
| gen_statem:start_link(?MODULE, [], []). |
| |
| %% gen_statem. |
| |
| callback_mode() -> |
| state_functions. |
| |
| init([]) -> |
| {ok, state_name, #state{}}. |
| |
| state_name(_EventType, _EventData, StateData) -> |
| {next_state, state_name, StateData}. |
| |
| handle_event(_EventType, _EventData, StateName, StateData) -> |
| {next_state, StateName, StateData}. |
| |
| terminate(_Reason, _StateName, _StateData) -> |
| ok. |
| |
| code_change(_OldVsn, StateName, StateData, _Extra) -> |
| {ok, StateName, StateData}. |
| endef |
| |
| define tpl_cowboy_loop |
| -module($(n)). |
| -behaviour(cowboy_loop_handler). |
| |
| -export([init/3]). |
| -export([info/3]). |
| -export([terminate/3]). |
| |
| -record(state, { |
| }). |
| |
| init(_, Req, _Opts) -> |
| {loop, Req, #state{}, 5000, hibernate}. |
| |
| info(_Info, Req, State) -> |
| {loop, Req, State, hibernate}. |
| |
| terminate(_Reason, _Req, _State) -> |
| ok. |
| endef |
| |
| define tpl_cowboy_rest |
| -module($(n)). |
| |
| -export([init/3]). |
| -export([content_types_provided/2]). |
| -export([get_html/2]). |
| |
| init(_, _Req, _Opts) -> |
| {upgrade, protocol, cowboy_rest}. |
| |
| content_types_provided(Req, State) -> |
| {[{{<<"text">>, <<"html">>, '*'}, get_html}], Req, State}. |
| |
| get_html(Req, State) -> |
| {<<"<html><body>This is REST!</body></html>">>, Req, State}. |
| endef |
| |
| define tpl_cowboy_ws |
| -module($(n)). |
| -behaviour(cowboy_websocket_handler). |
| |
| -export([init/3]). |
| -export([websocket_init/3]). |
| -export([websocket_handle/3]). |
| -export([websocket_info/3]). |
| -export([websocket_terminate/3]). |
| |
| -record(state, { |
| }). |
| |
| init(_, _, _) -> |
| {upgrade, protocol, cowboy_websocket}. |
| |
| websocket_init(_, Req, _Opts) -> |
| Req2 = cowboy_req:compact(Req), |
| {ok, Req2, #state{}}. |
| |
| websocket_handle({text, Data}, Req, State) -> |
| {reply, {text, Data}, Req, State}; |
| websocket_handle({binary, Data}, Req, State) -> |
| {reply, {binary, Data}, Req, State}; |
| websocket_handle(_Frame, Req, State) -> |
| {ok, Req, State}. |
| |
| websocket_info(_Info, Req, State) -> |
| {ok, Req, State}. |
| |
| websocket_terminate(_Reason, _Req, _State) -> |
| ok. |
| endef |
| |
| define tpl_ranch_protocol |
| -module($(n)). |
| -behaviour(ranch_protocol). |
| |
| -export([start_link/4]). |
| -export([init/4]). |
| |
| -type opts() :: []. |
| -export_type([opts/0]). |
| |
| -record(state, { |
| socket :: inet:socket(), |
| transport :: module() |
| }). |
| |
| start_link(Ref, Socket, Transport, Opts) -> |
| Pid = spawn_link(?MODULE, init, [Ref, Socket, Transport, Opts]), |
| {ok, Pid}. |
| |
| -spec init(ranch:ref(), inet:socket(), module(), opts()) -> ok. |
| init(Ref, Socket, Transport, _Opts) -> |
| ok = ranch:accept_ack(Ref), |
| loop(#state{socket=Socket, transport=Transport}). |
| |
| loop(State) -> |
| loop(State). |
| endef |
| |
| # Plugin-specific targets. |
| |
| ifndef WS |
| ifdef SP |
| WS = $(subst a,,a $(wordlist 1,$(SP),a a a a a a a a a a a a a a a a a a a a)) |
| else |
| WS = $(tab) |
| endif |
| endif |
| |
| bootstrap: |
| ifneq ($(wildcard src/),) |
| $(error Error: src/ directory already exists) |
| endif |
| $(eval p := $(PROJECT)) |
| $(if $(shell echo $p | LC_ALL=C grep -x "[a-z0-9_]*"),,\ |
| $(error Error: Invalid characters in the application name)) |
| $(eval n := $(PROJECT)_sup) |
| $(verbose) $(call core_render,bs_Makefile,Makefile) |
| $(verbose) echo "include erlang.mk" >> Makefile |
| $(verbose) mkdir src/ |
| ifdef LEGACY |
| $(verbose) $(call core_render,bs_appsrc,src/$(PROJECT).app.src) |
| endif |
| $(verbose) $(call core_render,bs_app,src/$(PROJECT)_app.erl) |
| $(verbose) $(call core_render,tpl_supervisor,src/$(PROJECT)_sup.erl) |
| |
| bootstrap-lib: |
| ifneq ($(wildcard src/),) |
| $(error Error: src/ directory already exists) |
| endif |
| $(eval p := $(PROJECT)) |
| $(if $(shell echo $p | LC_ALL=C grep -x "[a-z0-9_]*"),,\ |
| $(error Error: Invalid characters in the application name)) |
| $(verbose) $(call core_render,bs_Makefile,Makefile) |
| $(verbose) echo "include erlang.mk" >> Makefile |
| $(verbose) mkdir src/ |
| ifdef LEGACY |
| $(verbose) $(call core_render,bs_appsrc_lib,src/$(PROJECT).app.src) |
| endif |
| |
| bootstrap-rel: |
| ifneq ($(wildcard relx.config),) |
| $(error Error: relx.config already exists) |
| endif |
| ifneq ($(wildcard config/),) |
| $(error Error: config/ directory already exists) |
| endif |
| $(eval p := $(PROJECT)) |
| $(verbose) $(call core_render,bs_relx_config,relx.config) |
| $(verbose) mkdir config/ |
| $(verbose) $(call core_render,bs_sys_config,config/sys.config) |
| $(verbose) $(call core_render,bs_vm_args,config/vm.args) |
| $(verbose) awk '/^include erlang.mk/ && !ins {print "REL_DEPS += relx";ins=1};{print}' Makefile > Makefile.bak |
| $(verbose) mv Makefile.bak Makefile |
| |
| new-app: |
| ifndef in |
| $(error Usage: $(MAKE) new-app in=APP) |
| endif |
| ifneq ($(wildcard $(APPS_DIR)/$in),) |
| $(error Error: Application $in already exists) |
| endif |
| $(eval p := $(in)) |
| $(if $(shell echo $p | LC_ALL=C grep -x "[a-z0-9_]*"),,\ |
| $(error Error: Invalid characters in the application name)) |
| $(eval n := $(in)_sup) |
| $(verbose) mkdir -p $(APPS_DIR)/$p/src/ |
| $(verbose) $(call core_render,bs_apps_Makefile,$(APPS_DIR)/$p/Makefile) |
| ifdef LEGACY |
| $(verbose) $(call core_render,bs_appsrc,$(APPS_DIR)/$p/src/$p.app.src) |
| endif |
| $(verbose) $(call core_render,bs_app,$(APPS_DIR)/$p/src/$p_app.erl) |
| $(verbose) $(call core_render,tpl_supervisor,$(APPS_DIR)/$p/src/$p_sup.erl) |
| |
| new-lib: |
| ifndef in |
| $(error Usage: $(MAKE) new-lib in=APP) |
| endif |
| ifneq ($(wildcard $(APPS_DIR)/$in),) |
| $(error Error: Application $in already exists) |
| endif |
| $(eval p := $(in)) |
| $(if $(shell echo $p | LC_ALL=C grep -x "[a-z0-9_]*"),,\ |
| $(error Error: Invalid characters in the application name)) |
| $(verbose) mkdir -p $(APPS_DIR)/$p/src/ |
| $(verbose) $(call core_render,bs_apps_Makefile,$(APPS_DIR)/$p/Makefile) |
| ifdef LEGACY |
| $(verbose) $(call core_render,bs_appsrc_lib,$(APPS_DIR)/$p/src/$p.app.src) |
| endif |
| |
| new: |
| ifeq ($(wildcard src/)$(in),) |
| $(error Error: src/ directory does not exist) |
| endif |
| ifndef t |
| $(error Usage: $(MAKE) new t=TEMPLATE n=NAME [in=APP]) |
| endif |
| ifndef n |
| $(error Usage: $(MAKE) new t=TEMPLATE n=NAME [in=APP]) |
| endif |
| ifdef in |
| $(verbose) $(call core_render,tpl_$(t),$(APPS_DIR)/$(in)/src/$(n).erl) |
| else |
| $(verbose) $(call core_render,tpl_$(t),src/$(n).erl) |
| endif |
| |
| list-templates: |
| $(verbose) @echo Available templates: |
| $(verbose) printf " %s\n" $(sort $(patsubst tpl_%,%,$(filter tpl_%,$(.VARIABLES)))) |
| |
| # Copyright (c) 2014-2016, Loïc Hoguin <essen@ninenines.eu> |
| # This file is part of erlang.mk and subject to the terms of the ISC License. |
| |
| .PHONY: clean-c_src distclean-c_src-env |
| |
| # Configuration. |
| |
| C_SRC_DIR ?= $(CURDIR)/c_src |
| C_SRC_ENV ?= $(C_SRC_DIR)/env.mk |
| C_SRC_OUTPUT ?= $(CURDIR)/priv/$(PROJECT) |
| C_SRC_TYPE ?= shared |
| |
| # System type and C compiler/flags. |
| |
| ifeq ($(PLATFORM),msys2) |
| C_SRC_OUTPUT_EXECUTABLE_EXTENSION ?= .exe |
| C_SRC_OUTPUT_SHARED_EXTENSION ?= .dll |
| C_SRC_OUTPUT_STATIC_EXTENSION ?= .lib |
| else |
| C_SRC_OUTPUT_EXECUTABLE_EXTENSION ?= |
| C_SRC_OUTPUT_SHARED_EXTENSION ?= .so |
| C_SRC_OUTPUT_STATIC_EXTENSION ?= .a |
| endif |
| |
| ifeq ($(C_SRC_TYPE),shared) |
| C_SRC_OUTPUT_FILE = $(C_SRC_OUTPUT)$(C_SRC_OUTPUT_SHARED_EXTENSION) |
| else ifeq ($(C_SRC_TYPE),static) |
| C_SRC_OUTPUT_FILE = $(C_SRC_OUTPUT)$(C_SRC_OUTPUT_STATIC_EXTENSION) |
| else |
| C_SRC_OUTPUT_FILE = $(C_SRC_OUTPUT)$(C_SRC_OUTPUT_EXECUTABLE_EXTENSION) |
| endif |
| |
| RANLIB ?= ranlib |
| ARFLAGS ?= cr |
| |
| ifeq ($(PLATFORM),msys2) |
| # We hardcode the compiler used on MSYS2. The default CC=cc does |
| # not produce working code. The "gcc" MSYS2 package also doesn't. |
| CC = /mingw64/bin/gcc |
| export CC |
| CFLAGS ?= -O3 -std=c99 -finline-functions -Wall -Wmissing-prototypes |
| CXXFLAGS ?= -O3 -finline-functions -Wall |
| else ifeq ($(PLATFORM),darwin) |
| CC ?= cc |
| CFLAGS ?= -O3 -std=c99 -Wall -Wmissing-prototypes |
| CXXFLAGS ?= -O3 -Wall |
| LDFLAGS ?= -flat_namespace -undefined suppress |
| else ifeq ($(PLATFORM),freebsd) |
| CC ?= cc |
| CFLAGS ?= -O3 -std=c99 -finline-functions -Wall -Wmissing-prototypes |
| CXXFLAGS ?= -O3 -finline-functions -Wall |
| else ifeq ($(PLATFORM),linux) |
| CC ?= gcc |
| CFLAGS ?= -O3 -std=c99 -finline-functions -Wall -Wmissing-prototypes |
| CXXFLAGS ?= -O3 -finline-functions -Wall |
| endif |
| |
| ifneq ($(PLATFORM),msys2) |
| CFLAGS += -fPIC |
| CXXFLAGS += -fPIC |
| endif |
| |
| ifeq ($(C_SRC_TYPE),static) |
| CFLAGS += -DSTATIC_ERLANG_NIF=1 |
| CXXFLAGS += -DSTATIC_ERLANG_NIF=1 |
| endif |
| |
| CFLAGS += -I"$(ERTS_INCLUDE_DIR)" -I"$(ERL_INTERFACE_INCLUDE_DIR)" |
| CXXFLAGS += -I"$(ERTS_INCLUDE_DIR)" -I"$(ERL_INTERFACE_INCLUDE_DIR)" |
| |
| LDLIBS += -L"$(ERL_INTERFACE_LIB_DIR)" -lei |
| |
| # Verbosity. |
| |
| c_verbose_0 = @echo " C " $(filter-out $(notdir $(MAKEFILE_LIST) $(C_SRC_ENV)),$(^F)); |
| c_verbose = $(c_verbose_$(V)) |
| |
| cpp_verbose_0 = @echo " CPP " $(filter-out $(notdir $(MAKEFILE_LIST) $(C_SRC_ENV)),$(^F)); |
| cpp_verbose = $(cpp_verbose_$(V)) |
| |
| link_verbose_0 = @echo " LD " $(@F); |
| link_verbose = $(link_verbose_$(V)) |
| |
| ar_verbose_0 = @echo " AR " $(@F); |
| ar_verbose = $(ar_verbose_$(V)) |
| |
| ranlib_verbose_0 = @echo " RANLIB" $(@F); |
| ranlib_verbose = $(ranlib_verbose_$(V)) |
| |
| # Targets. |
| |
| ifeq ($(wildcard $(C_SRC_DIR)),) |
| else ifneq ($(wildcard $(C_SRC_DIR)/Makefile),) |
| app:: app-c_src |
| |
| test-build:: app-c_src |
| |
| app-c_src: |
| $(MAKE) -C $(C_SRC_DIR) |
| |
| clean:: |
| $(MAKE) -C $(C_SRC_DIR) clean |
| |
| else |
| |
| ifeq ($(SOURCES),) |
| SOURCES := $(sort $(foreach pat,*.c *.C *.cc *.cpp,$(call core_find,$(C_SRC_DIR)/,$(pat)))) |
| endif |
| OBJECTS = $(addsuffix .o, $(basename $(SOURCES))) |
| |
| COMPILE_C = $(c_verbose) $(CC) $(CFLAGS) $(CPPFLAGS) -c |
| COMPILE_CPP = $(cpp_verbose) $(CXX) $(CXXFLAGS) $(CPPFLAGS) -c |
| |
| app:: $(C_SRC_ENV) $(C_SRC_OUTPUT_FILE) |
| |
| test-build:: $(C_SRC_ENV) $(C_SRC_OUTPUT_FILE) |
| |
| ifneq ($(C_SRC_TYPE),static) |
| $(C_SRC_OUTPUT_FILE): $(OBJECTS) |
| $(verbose) mkdir -p $(dir $@) |
| $(link_verbose) $(CC) $(OBJECTS) \ |
| $(LDFLAGS) $(if $(filter $(C_SRC_TYPE),shared),-shared) $(LDLIBS) \ |
| -o $(C_SRC_OUTPUT_FILE) |
| else |
| $(C_SRC_OUTPUT_FILE): $(OBJECTS) |
| $(verbose) mkdir -p $(dir $@) |
| $(ar_verbose) $(AR) $(ARFLAGS) $(C_SRC_OUTPUT_FILE) $(OBJECTS) |
| $(ranlib_verbose) $(RANLIB) $(C_SRC_OUTPUT_FILE) |
| endif |
| |
| |
| $(OBJECTS): $(MAKEFILE_LIST) $(C_SRC_ENV) |
| |
| %.o: %.c |
| $(COMPILE_C) $(OUTPUT_OPTION) $< |
| |
| %.o: %.cc |
| $(COMPILE_CPP) $(OUTPUT_OPTION) $< |
| |
| %.o: %.C |
| $(COMPILE_CPP) $(OUTPUT_OPTION) $< |
| |
| %.o: %.cpp |
| $(COMPILE_CPP) $(OUTPUT_OPTION) $< |
| |
| clean:: clean-c_src |
| |
| clean-c_src: |
| $(gen_verbose) rm -f $(C_SRC_OUTPUT_FILE) $(OBJECTS) |
| |
| endif |
| |
| ifneq ($(wildcard $(C_SRC_DIR)),) |
| ERL_ERTS_DIR = $(shell $(ERL) -eval 'io:format("~s~n", [code:lib_dir(erts)]), halt().') |
| |
| $(C_SRC_ENV): |
| $(verbose) $(ERL) -eval "file:write_file(\"$(call core_native_path,$(C_SRC_ENV))\", \ |
| io_lib:format( \ |
| \"# Generated by Erlang.mk. Edit at your own risk!~n~n\" \ |
| \"ERTS_INCLUDE_DIR ?= ~s/erts-~s/include/~n\" \ |
| \"ERL_INTERFACE_INCLUDE_DIR ?= ~s~n\" \ |
| \"ERL_INTERFACE_LIB_DIR ?= ~s~n\" \ |
| \"ERTS_DIR ?= $(ERL_ERTS_DIR)~n\", \ |
| [code:root_dir(), erlang:system_info(version), \ |
| code:lib_dir(erl_interface, include), \ |
| code:lib_dir(erl_interface, lib)])), \ |
| halt()." |
| |
| distclean:: distclean-c_src-env |
| |
| distclean-c_src-env: |
| $(gen_verbose) rm -f $(C_SRC_ENV) |
| |
| -include $(C_SRC_ENV) |
| |
| ifneq ($(ERL_ERTS_DIR),$(ERTS_DIR)) |
| $(shell rm -f $(C_SRC_ENV)) |
| endif |
| endif |
| |
| # Templates. |
| |
| define bs_c_nif |
| #include "erl_nif.h" |
| |
| static int loads = 0; |
| |
| static int load(ErlNifEnv* env, void** priv_data, ERL_NIF_TERM load_info) |
| { |
| /* Initialize private data. */ |
| *priv_data = NULL; |
| |
| loads++; |
| |
| return 0; |
| } |
| |
| static int upgrade(ErlNifEnv* env, void** priv_data, void** old_priv_data, ERL_NIF_TERM load_info) |
| { |
| /* Convert the private data to the new version. */ |
| *priv_data = *old_priv_data; |
| |
| loads++; |
| |
| return 0; |
| } |
| |
| static void unload(ErlNifEnv* env, void* priv_data) |
| { |
| if (loads == 1) { |
| /* Destroy the private data. */ |
| } |
| |
| loads--; |
| } |
| |
| static ERL_NIF_TERM hello(ErlNifEnv* env, int argc, const ERL_NIF_TERM argv[]) |
| { |
| if (enif_is_atom(env, argv[0])) { |
| return enif_make_tuple2(env, |
| enif_make_atom(env, "hello"), |
| argv[0]); |
| } |
| |
| return enif_make_tuple2(env, |
| enif_make_atom(env, "error"), |
| enif_make_atom(env, "badarg")); |
| } |
| |
| static ErlNifFunc nif_funcs[] = { |
| {"hello", 1, hello} |
| }; |
| |
| ERL_NIF_INIT($n, nif_funcs, load, NULL, upgrade, unload) |
| endef |
| |
| define bs_erl_nif |
| -module($n). |
| |
| -export([hello/1]). |
| |
| -on_load(on_load/0). |
| on_load() -> |
| PrivDir = case code:priv_dir(?MODULE) of |
| {error, _} -> |
| AppPath = filename:dirname(filename:dirname(code:which(?MODULE))), |
| filename:join(AppPath, "priv"); |
| Path -> |
| Path |
| end, |
| erlang:load_nif(filename:join(PrivDir, atom_to_list(?MODULE)), 0). |
| |
| hello(_) -> |
| erlang:nif_error({not_loaded, ?MODULE}). |
| endef |
| |
| new-nif: |
| ifneq ($(wildcard $(C_SRC_DIR)/$n.c),) |
| $(error Error: $(C_SRC_DIR)/$n.c already exists) |
| endif |
| ifneq ($(wildcard src/$n.erl),) |
| $(error Error: src/$n.erl already exists) |
| endif |
| ifndef n |
| $(error Usage: $(MAKE) new-nif n=NAME [in=APP]) |
| endif |
| ifdef in |
| $(verbose) $(MAKE) -C $(APPS_DIR)/$(in)/ new-nif n=$n in= |
| else |
| $(verbose) mkdir -p $(C_SRC_DIR) src/ |
| $(verbose) $(call core_render,bs_c_nif,$(C_SRC_DIR)/$n.c) |
| $(verbose) $(call core_render,bs_erl_nif,src/$n.erl) |
| endif |
| |
| # Copyright (c) 2015-2017, Loïc Hoguin <essen@ninenines.eu> |
| # This file is part of erlang.mk and subject to the terms of the ISC License. |
| |
| .PHONY: ci ci-prepare ci-setup |
| |
| CI_OTP ?= |
| |
| ifeq ($(strip $(CI_OTP)),) |
| ci:: |
| else |
| |
| ci:: $(addprefix ci-,$(CI_OTP)) |
| |
| ci-prepare: $(addprefix ci-prepare-,$(CI_OTP)) |
| |
| ci-setup:: |
| $(verbose) : |
| |
| ci-extra:: |
| $(verbose) : |
| |
| ci_verbose_0 = @echo " CI " $(1); |
| ci_verbose = $(ci_verbose_$(V)) |
| |
| define ci_target |
| ci-prepare-$1: $(KERL_INSTALL_DIR)/$2 |
| $(verbose) : |
| |
| ci-$1: ci-prepare-$1 |
| $(verbose) $(MAKE) --no-print-directory clean |
| $(ci_verbose) \ |
| PATH="$(KERL_INSTALL_DIR)/$2/bin:$(PATH)" \ |
| CI_OTP_RELEASE="$1" \ |
| CT_OPTS="-label $1" \ |
| CI_VM="$3" \ |
| $(MAKE) ci-setup tests |
| $(verbose) $(MAKE) --no-print-directory ci-extra |
| endef |
| |
| $(foreach otp,$(CI_OTP),$(eval $(call ci_target,$(otp),$(otp),otp))) |
| |
| $(foreach otp,$(filter-out $(ERLANG_OTP),$(CI_OTP)),$(eval $(call kerl_otp_target,$(otp)))) |
| |
| help:: |
| $(verbose) printf "%s\n" "" \ |
| "Continuous Integration targets:" \ |
| " ci Run '$(MAKE) tests' on all configured Erlang versions." \ |
| "" \ |
| "The CI_OTP variable must be defined with the Erlang versions" \ |
| "that must be tested. For example: CI_OTP = OTP-17.3.4 OTP-17.5.3" |
| |
| endif |
| |
| # Copyright (c) 2020, Loïc Hoguin <essen@ninenines.eu> |
| # This file is part of erlang.mk and subject to the terms of the ISC License. |
| |
| ifdef CONCUERROR_TESTS |
| |
| .PHONY: concuerror distclean-concuerror |
| |
| # Configuration |
| |
| CONCUERROR_LOGS_DIR ?= $(CURDIR)/logs |
| CONCUERROR_OPTS ?= |
| |
| # Core targets. |
| |
| check:: concuerror |
| |
| ifndef KEEP_LOGS |
| distclean:: distclean-concuerror |
| endif |
| |
| # Plugin-specific targets. |
| |
| $(ERLANG_MK_TMP)/Concuerror/bin/concuerror: | $(ERLANG_MK_TMP) |
| $(verbose) git clone https://github.com/parapluu/Concuerror $(ERLANG_MK_TMP)/Concuerror |
| $(verbose) $(MAKE) -C $(ERLANG_MK_TMP)/Concuerror |
| |
| $(CONCUERROR_LOGS_DIR): |
| $(verbose) mkdir -p $(CONCUERROR_LOGS_DIR) |
| |
| define concuerror_html_report |
| <!DOCTYPE html> |
| <html lang="en"> |
| <head> |
| <meta charset="utf-8"> |
| <title>Concuerror HTML report</title> |
| </head> |
| <body> |
| <h1>Concuerror HTML report</h1> |
| <p>Generated on $(concuerror_date)</p> |
| <ul> |
| $(foreach t,$(concuerror_targets),<li><a href="$(t).txt">$(t)</a></li>) |
| </ul> |
| </body> |
| </html> |
| endef |
| |
| concuerror: $(addprefix concuerror-,$(subst :,-,$(CONCUERROR_TESTS))) |
| $(eval concuerror_date := $(shell date)) |
| $(eval concuerror_targets := $^) |
| $(verbose) $(call core_render,concuerror_html_report,$(CONCUERROR_LOGS_DIR)/concuerror.html) |
| |
| define concuerror_target |
| .PHONY: concuerror-$1-$2 |
| |
| concuerror-$1-$2: test-build | $(ERLANG_MK_TMP)/Concuerror/bin/concuerror $(CONCUERROR_LOGS_DIR) |
| $(ERLANG_MK_TMP)/Concuerror/bin/concuerror \ |
| --pa $(CURDIR)/ebin --pa $(TEST_DIR) \ |
| -o $(CONCUERROR_LOGS_DIR)/concuerror-$1-$2.txt \ |
| $$(CONCUERROR_OPTS) -m $1 -t $2 |
| endef |
| |
| $(foreach test,$(CONCUERROR_TESTS),$(eval $(call concuerror_target,$(firstword $(subst :, ,$(test))),$(lastword $(subst :, ,$(test)))))) |
| |
| distclean-concuerror: |
| $(gen_verbose) rm -rf $(CONCUERROR_LOGS_DIR) |
| |
| endif |
| |
| # Copyright (c) 2013-2016, Loïc Hoguin <essen@ninenines.eu> |
| # This file is part of erlang.mk and subject to the terms of the ISC License. |
| |
| .PHONY: ct apps-ct distclean-ct |
| |
| # Configuration. |
| |
| CT_OPTS ?= |
| |
| ifneq ($(wildcard $(TEST_DIR)),) |
| ifndef CT_SUITES |
| CT_SUITES := $(sort $(subst _SUITE.erl,,$(notdir $(call core_find,$(TEST_DIR)/,*_SUITE.erl)))) |
| endif |
| endif |
| CT_SUITES ?= |
| CT_LOGS_DIR ?= $(CURDIR)/logs |
| |
| # Core targets. |
| |
| tests:: ct |
| |
| ifndef KEEP_LOGS |
| distclean:: distclean-ct |
| endif |
| |
| help:: |
| $(verbose) printf "%s\n" "" \ |
| "Common_test targets:" \ |
| " ct Run all the common_test suites for this project" \ |
| "" \ |
| "All your common_test suites have their associated targets." \ |
| "A suite named http_SUITE can be ran using the ct-http target." |
| |
| # Plugin-specific targets. |
| |
| CT_RUN = ct_run \ |
| -no_auto_compile \ |
| -noinput \ |
| -pa $(CURDIR)/ebin $(TEST_DIR) \ |
| -dir $(TEST_DIR) \ |
| -logdir $(CT_LOGS_DIR) |
| |
| ifeq ($(CT_SUITES),) |
| ct: $(if $(IS_APP)$(ROOT_DIR),,apps-ct) |
| else |
| # We do not run tests if we are in an apps/* with no test directory. |
| ifneq ($(IS_APP)$(wildcard $(TEST_DIR)),1) |
| ct: test-build $(if $(IS_APP)$(ROOT_DIR),,apps-ct) |
| $(verbose) mkdir -p $(CT_LOGS_DIR) |
| $(gen_verbose) $(CT_RUN) -sname ct_$(PROJECT) -suite $(addsuffix _SUITE,$(CT_SUITES)) $(CT_OPTS) |
| endif |
| endif |
| |
| ifneq ($(ALL_APPS_DIRS),) |
| define ct_app_target |
| apps-ct-$1: test-build |
| $$(MAKE) -C $1 ct IS_APP=1 |
| endef |
| |
| $(foreach app,$(ALL_APPS_DIRS),$(eval $(call ct_app_target,$(app)))) |
| |
| apps-ct: $(addprefix apps-ct-,$(ALL_APPS_DIRS)) |
| endif |
| |
| ifdef t |
| ifeq (,$(findstring :,$t)) |
| CT_EXTRA = -group $t |
| else |
| t_words = $(subst :, ,$t) |
| CT_EXTRA = -group $(firstword $(t_words)) -case $(lastword $(t_words)) |
| endif |
| else |
| ifdef c |
| CT_EXTRA = -case $c |
| else |
| CT_EXTRA = |
| endif |
| endif |
| |
| define ct_suite_target |
| ct-$1: test-build |
| $$(verbose) mkdir -p $$(CT_LOGS_DIR) |
| $$(gen_verbose_esc) $$(CT_RUN) -sname ct_$$(PROJECT) -suite $$(addsuffix _SUITE,$1) $$(CT_EXTRA) $$(CT_OPTS) |
| endef |
| |
| $(foreach test,$(CT_SUITES),$(eval $(call ct_suite_target,$(test)))) |
| |
| distclean-ct: |
| $(gen_verbose) rm -rf $(CT_LOGS_DIR) |
| |
| # Copyright (c) 2013-2016, Loïc Hoguin <essen@ninenines.eu> |
| # This file is part of erlang.mk and subject to the terms of the ISC License. |
| |
| .PHONY: plt distclean-plt dialyze |
| |
| # Configuration. |
| |
| DIALYZER_PLT ?= $(CURDIR)/.$(PROJECT).plt |
| export DIALYZER_PLT |
| |
| PLT_APPS ?= |
| DIALYZER_DIRS ?= --src -r $(wildcard src) $(ALL_APPS_DIRS) |
| DIALYZER_OPTS ?= -Werror_handling -Wunmatched_returns # -Wunderspecs |
| DIALYZER_PLT_OPTS ?= |
| |
| # Core targets. |
| |
| check:: dialyze |
| |
| distclean:: distclean-plt |
| |
| help:: |
| $(verbose) printf "%s\n" "" \ |
| "Dialyzer targets:" \ |
| " plt Build a PLT file for this project" \ |
| " dialyze Analyze the project using Dialyzer" |
| |
| # Plugin-specific targets. |
| |
| define filter_opts.erl |
| Opts = init:get_plain_arguments(), |
| {Filtered, _} = lists:foldl(fun |
| (O, {Os, true}) -> {[O|Os], false}; |
| (O = "-D", {Os, _}) -> {[O|Os], true}; |
| (O = [\\$$-, \\$$D, _ | _], {Os, _}) -> {[O|Os], false}; |
| (O = "-I", {Os, _}) -> {[O|Os], true}; |
| (O = [\\$$-, \\$$I, _ | _], {Os, _}) -> {[O|Os], false}; |
| (O = "-pa", {Os, _}) -> {[O|Os], true}; |
| (_, Acc) -> Acc |
| end, {[], false}, Opts), |
| io:format("~s~n", [string:join(lists:reverse(Filtered), " ")]), |
| halt(). |
| endef |
| |
| # DIALYZER_PLT is a variable understood directly by Dialyzer. |
| # |
| # We append the path to erts at the end of the PLT. This works |
| # because the PLT file is in the external term format and the |
| # function binary_to_term/1 ignores any trailing data. |
| $(DIALYZER_PLT): deps app |
| $(eval DEPS_LOG := $(shell test -f $(ERLANG_MK_TMP)/deps.log && \ |
| while read p; do test -d $$p/ebin && echo $$p/ebin; done <$(ERLANG_MK_TMP)/deps.log)) |
| $(verbose) dialyzer --build_plt $(DIALYZER_PLT_OPTS) --apps \ |
| erts kernel stdlib $(PLT_APPS) $(OTP_DEPS) $(LOCAL_DEPS) $(DEPS_LOG) || test $$? -eq 2 |
| $(verbose) $(ERL) -eval 'io:format("~n~s~n", [code:lib_dir(erts)]), halt().' >> $@ |
| |
| plt: $(DIALYZER_PLT) |
| |
| distclean-plt: |
| $(gen_verbose) rm -f $(DIALYZER_PLT) |
| |
| ifneq ($(wildcard $(DIALYZER_PLT)),) |
| dialyze: $(if $(filter --src,$(DIALYZER_DIRS)),,deps app) |
| $(verbose) if ! tail -n1 $(DIALYZER_PLT) | \ |
| grep -q "^`$(ERL) -eval 'io:format("~s", [code:lib_dir(erts)]), halt().'`$$"; then \ |
| rm $(DIALYZER_PLT); \ |
| $(MAKE) plt; \ |
| fi |
| else |
| dialyze: $(DIALYZER_PLT) |
| endif |
| $(verbose) dialyzer `$(ERL) \ |
| -eval "$(subst $(newline),,$(call escape_dquotes,$(call filter_opts.erl)))" \ |
| -extra $(ERLC_OPTS)` $(DIALYZER_DIRS) $(DIALYZER_OPTS) $(if $(wildcard ebin/),-pa ebin/) |
| |
| # Copyright (c) 2013-2016, Loïc Hoguin <essen@ninenines.eu> |
| # This file is part of erlang.mk and subject to the terms of the ISC License. |
| |
| .PHONY: distclean-edoc edoc |
| |
| # Configuration. |
| |
| EDOC_OPTS ?= |
| EDOC_SRC_DIRS ?= |
| EDOC_OUTPUT ?= doc |
| |
| define edoc.erl |
| SrcPaths = lists:foldl(fun(P, Acc) -> |
| filelib:wildcard(atom_to_list(P) ++ "/{src,c_src}") |
| ++ lists:filter(fun(D) -> |
| filelib:is_dir(D) |
| end, filelib:wildcard(atom_to_list(P) ++ "/{src,c_src}/**")) |
| ++ Acc |
| end, [], [$(call comma_list,$(patsubst %,'%',$(call core_native_path,$(EDOC_SRC_DIRS))))]), |
| DefaultOpts = [{dir, "$(EDOC_OUTPUT)"}, {source_path, SrcPaths}, {subpackages, false}], |
| edoc:application($(1), ".", [$(2)] ++ DefaultOpts), |
| halt(0). |
| endef |
| |
| # Core targets. |
| |
| ifneq ($(strip $(EDOC_SRC_DIRS)$(wildcard doc/overview.edoc)),) |
| docs:: edoc |
| endif |
| |
| distclean:: distclean-edoc |
| |
| # Plugin-specific targets. |
| |
| edoc: distclean-edoc doc-deps |
| $(gen_verbose) $(call erlang,$(call edoc.erl,$(PROJECT),$(EDOC_OPTS))) |
| |
| distclean-edoc: |
| $(gen_verbose) rm -f $(EDOC_OUTPUT)/*.css $(EDOC_OUTPUT)/*.html $(EDOC_OUTPUT)/*.png $(EDOC_OUTPUT)/edoc-info |
| |
| # Copyright (c) 2013-2016, Loïc Hoguin <essen@ninenines.eu> |
| # This file is part of erlang.mk and subject to the terms of the ISC License. |
| |
| # Configuration. |
| |
| DTL_FULL_PATH ?= |
| DTL_PATH ?= templates/ |
| DTL_PREFIX ?= |
| DTL_SUFFIX ?= _dtl |
| DTL_OPTS ?= |
| |
| # Verbosity. |
| |
| dtl_verbose_0 = @echo " DTL " $(filter %.dtl,$(?F)); |
| dtl_verbose = $(dtl_verbose_$(V)) |
| |
| # Core targets. |
| |
| DTL_PATH := $(abspath $(DTL_PATH)) |
| DTL_FILES := $(sort $(call core_find,$(DTL_PATH),*.dtl)) |
| |
| ifneq ($(DTL_FILES),) |
| |
| DTL_NAMES = $(addprefix $(DTL_PREFIX),$(addsuffix $(DTL_SUFFIX),$(DTL_FILES:$(DTL_PATH)/%.dtl=%))) |
| DTL_MODULES = $(if $(DTL_FULL_PATH),$(subst /,_,$(DTL_NAMES)),$(notdir $(DTL_NAMES))) |
| BEAM_FILES += $(addsuffix .beam,$(addprefix ebin/,$(DTL_MODULES))) |
| |
| ifneq ($(words $(DTL_FILES)),0) |
| # Rebuild templates when the Makefile changes. |
| $(ERLANG_MK_TMP)/last-makefile-change-erlydtl: $(MAKEFILE_LIST) | $(ERLANG_MK_TMP) |
| $(verbose) if test -f $@; then \ |
| touch $(DTL_FILES); \ |
| fi |
| $(verbose) touch $@ |
| |
| ebin/$(PROJECT).app:: $(ERLANG_MK_TMP)/last-makefile-change-erlydtl |
| endif |
| |
| define erlydtl_compile.erl |
| [begin |
| Module0 = case "$(strip $(DTL_FULL_PATH))" of |
| "" -> |
| filename:basename(F, ".dtl"); |
| _ -> |
| "$(call core_native_path,$(DTL_PATH))/" ++ F2 = filename:rootname(F, ".dtl"), |
| re:replace(F2, "/", "_", [{return, list}, global]) |
| end, |
| Module = list_to_atom("$(DTL_PREFIX)" ++ string:to_lower(Module0) ++ "$(DTL_SUFFIX)"), |
| case erlydtl:compile(F, Module, [$(DTL_OPTS)] ++ [{out_dir, "ebin/"}, return_errors]) of |
| ok -> ok; |
| {ok, _} -> ok |
| end |
| end || F <- string:tokens("$(1)", " ")], |
| halt(). |
| endef |
| |
| ebin/$(PROJECT).app:: $(DTL_FILES) | ebin/ |
| $(if $(strip $?),\ |
| $(dtl_verbose) $(call erlang,$(call erlydtl_compile.erl,$(call core_native_path,$?)),\ |
| -pa ebin/)) |
| |
| endif |
| |
| # Copyright (c) 2016, Loïc Hoguin <essen@ninenines.eu> |
| # Copyright (c) 2014, Dave Cottlehuber <dch@skunkwerks.at> |
| # This file is part of erlang.mk and subject to the terms of the ISC License. |
| |
| .PHONY: distclean-escript escript escript-zip |
| |
| # Configuration. |
| |
| ESCRIPT_NAME ?= $(PROJECT) |
| ESCRIPT_FILE ?= $(ESCRIPT_NAME) |
| |
| ESCRIPT_SHEBANG ?= /usr/bin/env escript |
| ESCRIPT_COMMENT ?= This is an -*- erlang -*- file |
| ESCRIPT_EMU_ARGS ?= -escript main $(ESCRIPT_NAME) |
| |
| ESCRIPT_ZIP ?= 7z a -tzip -mx=9 -mtc=off $(if $(filter-out 0,$(V)),,> /dev/null) |
| ESCRIPT_ZIP_FILE ?= $(ERLANG_MK_TMP)/escript.zip |
| |
| # Core targets. |
| |
| distclean:: distclean-escript |
| |
| help:: |
| $(verbose) printf "%s\n" "" \ |
| "Escript targets:" \ |
| " escript Build an executable escript archive" \ |
| |
| # Plugin-specific targets. |
| |
| escript-zip:: FULL=1 |
| escript-zip:: deps app |
| $(verbose) mkdir -p $(dir $(abspath $(ESCRIPT_ZIP_FILE))) |
| $(verbose) rm -f $(abspath $(ESCRIPT_ZIP_FILE)) |
| $(gen_verbose) cd .. && $(ESCRIPT_ZIP) $(abspath $(ESCRIPT_ZIP_FILE)) $(PROJECT)/ebin/* |
| ifneq ($(DEPS),) |
| $(verbose) cd $(DEPS_DIR) && $(ESCRIPT_ZIP) $(abspath $(ESCRIPT_ZIP_FILE)) \ |
| $(subst $(DEPS_DIR)/,,$(addsuffix /*,$(wildcard \ |
| $(addsuffix /ebin,$(shell cat $(ERLANG_MK_TMP)/deps.log))))) |
| endif |
| |
| escript:: escript-zip |
| $(gen_verbose) printf "%s\n" \ |
| "#!$(ESCRIPT_SHEBANG)" \ |
| "%% $(ESCRIPT_COMMENT)" \ |
| "%%! $(ESCRIPT_EMU_ARGS)" > $(ESCRIPT_FILE) |
| $(verbose) cat $(abspath $(ESCRIPT_ZIP_FILE)) >> $(ESCRIPT_FILE) |
| $(verbose) chmod +x $(ESCRIPT_FILE) |
| |
| distclean-escript: |
| $(gen_verbose) rm -f $(ESCRIPT_FILE) $(abspath $(ESCRIPT_ZIP_FILE)) |
| |
| # Copyright (c) 2015-2016, Loïc Hoguin <essen@ninenines.eu> |
| # Copyright (c) 2014, Enrique Fernandez <enrique.fernandez@erlang-solutions.com> |
| # This file is contributed to erlang.mk and subject to the terms of the ISC License. |
| |
| .PHONY: eunit apps-eunit |
| |
| # Configuration |
| |
| EUNIT_OPTS ?= |
| EUNIT_ERL_OPTS ?= |
| EUNIT_TEST_SPEC ?= $1 |
| |
| # Core targets. |
| |
| tests:: eunit |
| |
| help:: |
| $(verbose) printf "%s\n" "" \ |
| "EUnit targets:" \ |
| " eunit Run all the EUnit tests for this project" |
| |
| # Plugin-specific targets. |
| |
| define eunit.erl |
| $(call cover.erl) |
| CoverSetup(), |
| case eunit:test($(call EUNIT_TEST_SPEC,$1), [$(EUNIT_OPTS)]) of |
| ok -> ok; |
| error -> halt(2) |
| end, |
| CoverExport("$(call core_native_path,$(COVER_DATA_DIR))/eunit.coverdata"), |
| halt() |
| endef |
| |
| EUNIT_ERL_OPTS += -pa $(TEST_DIR) $(CURDIR)/ebin |
| |
| ifdef t |
| ifeq (,$(findstring :,$(t))) |
| eunit: test-build cover-data-dir |
| $(gen_verbose) $(call erlang,$(call eunit.erl,['$(t)']),$(EUNIT_ERL_OPTS)) |
| else |
| eunit: test-build cover-data-dir |
| $(gen_verbose) $(call erlang,$(call eunit.erl,fun $(t)/0),$(EUNIT_ERL_OPTS)) |
| endif |
| else |
| EUNIT_EBIN_MODS = $(notdir $(basename $(ERL_FILES) $(BEAM_FILES))) |
| EUNIT_TEST_MODS = $(notdir $(basename $(call core_find,$(TEST_DIR)/,*.erl))) |
| |
| EUNIT_MODS = $(foreach mod,$(EUNIT_EBIN_MODS) $(filter-out \ |
| $(patsubst %,%_tests,$(EUNIT_EBIN_MODS)),$(EUNIT_TEST_MODS)),'$(mod)') |
| |
| eunit: test-build $(if $(IS_APP)$(ROOT_DIR),,apps-eunit) cover-data-dir |
| ifneq ($(wildcard src/ $(TEST_DIR)),) |
| $(gen_verbose) $(call erlang,$(call eunit.erl,[$(call comma_list,$(EUNIT_MODS))]),$(EUNIT_ERL_OPTS)) |
| endif |
| |
| ifneq ($(ALL_APPS_DIRS),) |
| apps-eunit: test-build |
| $(verbose) eunit_retcode=0 ; for app in $(ALL_APPS_DIRS); do $(MAKE) -C $$app eunit IS_APP=1; \ |
| [ $$? -ne 0 ] && eunit_retcode=1 ; done ; \ |
| exit $$eunit_retcode |
| endif |
| endif |
| |
| # Copyright (c) 2020, Loïc Hoguin <essen@ninenines.eu> |
| # This file is part of erlang.mk and subject to the terms of the ISC License. |
| |
| HEX_CORE_GIT ?= https://github.com/hexpm/hex_core |
| HEX_CORE_COMMIT ?= v0.7.0 |
| |
| PACKAGES += hex_core |
| pkg_hex_core_name = hex_core |
| pkg_hex_core_description = Reference implementation of Hex specifications |
| pkg_hex_core_homepage = $(HEX_CORE_GIT) |
| pkg_hex_core_fetch = git |
| pkg_hex_core_repo = $(HEX_CORE_GIT) |
| pkg_hex_core_commit = $(HEX_CORE_COMMIT) |
| |
| # We automatically depend on hex_core when the project isn't already. |
| $(if $(filter hex_core,$(DEPS) $(BUILD_DEPS) $(DOC_DEPS) $(REL_DEPS) $(TEST_DEPS)),,\ |
| $(eval $(call dep_target,hex_core))) |
| |
| hex-core: $(DEPS_DIR)/hex_core |
| $(verbose) if [ ! -e $(DEPS_DIR)/hex_core/ebin/dep_built ]; then \ |
| $(MAKE) -C $(DEPS_DIR)/hex_core IS_DEP=1; \ |
| touch $(DEPS_DIR)/hex_core/ebin/dep_built; \ |
| fi |
| |
| # @todo This must also apply to fetching. |
| HEX_CONFIG ?= |
| |
| define hex_config.erl |
| begin |
| Config0 = hex_core:default_config(), |
| Config0$(HEX_CONFIG) |
| end |
| endef |
| |
| define hex_user_create.erl |
| {ok, _} = application:ensure_all_started(ssl), |
| {ok, _} = application:ensure_all_started(inets), |
| Config = $(hex_config.erl), |
| case hex_api_user:create(Config, <<"$(strip $1)">>, <<"$(strip $2)">>, <<"$(strip $3)">>) of |
| {ok, {201, _, #{<<"email">> := Email, <<"url">> := URL, <<"username">> := Username}}} -> |
| io:format("User ~s (~s) created at ~s~n" |
| "Please check your inbox for a confirmation email.~n" |
| "You must confirm before you are allowed to publish packages.~n", |
| [Username, Email, URL]), |
| halt(0); |
| {ok, {Status, _, Errors}} -> |
| io:format("Error ~b: ~0p~n", [Status, Errors]), |
| halt(80) |
| end |
| endef |
| |
| # The $(info ) call inserts a new line after the password prompt. |
| hex-user-create: hex-core |
| $(if $(HEX_USERNAME),,$(eval HEX_USERNAME := $(shell read -p "Username: " username; echo $$username))) |
| $(if $(HEX_PASSWORD),,$(eval HEX_PASSWORD := $(shell stty -echo; read -p "Password: " password; stty echo; echo $$password) $(info ))) |
| $(if $(HEX_EMAIL),,$(eval HEX_EMAIL := $(shell read -p "Email: " email; echo $$email))) |
| $(gen_verbose) $(call erlang,$(call hex_user_create.erl,$(HEX_USERNAME),$(HEX_PASSWORD),$(HEX_EMAIL))) |
| |
| define hex_key_add.erl |
| {ok, _} = application:ensure_all_started(ssl), |
| {ok, _} = application:ensure_all_started(inets), |
| Config = $(hex_config.erl), |
| ConfigF = Config#{api_key => iolist_to_binary([<<"Basic ">>, base64:encode(<<"$(strip $1):$(strip $2)">>)])}, |
| Permissions = [ |
| case string:split(P, <<":">>) of |
| [D] -> #{domain => D}; |
| [D, R] -> #{domain => D, resource => R} |
| end |
| || P <- string:split(<<"$(strip $4)">>, <<",">>, all)], |
| case hex_api_key:add(ConfigF, <<"$(strip $3)">>, Permissions) of |
| {ok, {201, _, #{<<"secret">> := Secret}}} -> |
| io:format("Key ~s created for user ~s~nSecret: ~s~n" |
| "Please store the secret in a secure location, such as a password store.~n" |
| "The secret will be requested for most Hex-related operations.~n", |
| [<<"$(strip $3)">>, <<"$(strip $1)">>, Secret]), |
| halt(0); |
| {ok, {Status, _, Errors}} -> |
| io:format("Error ~b: ~0p~n", [Status, Errors]), |
| halt(81) |
| end |
| endef |
| |
| hex-key-add: hex-core |
| $(if $(HEX_USERNAME),,$(eval HEX_USERNAME := $(shell read -p "Username: " username; echo $$username))) |
| $(if $(HEX_PASSWORD),,$(eval HEX_PASSWORD := $(shell stty -echo; read -p "Password: " password; stty echo; echo $$password) $(info ))) |
| $(gen_verbose) $(call erlang,$(call hex_key_add.erl,$(HEX_USERNAME),$(HEX_PASSWORD),\ |
| $(if $(name),$(name),$(shell hostname)-erlang-mk),\ |
| $(if $(perm),$(perm),api))) |
| |
| HEX_TARBALL_EXTRA_METADATA ?= |
| |
| # @todo Check that we can += files |
| HEX_TARBALL_FILES ?= \ |
| $(wildcard early-plugins.mk) \ |
| $(wildcard ebin/$(PROJECT).app) \ |
| $(wildcard ebin/$(PROJECT).appup) \ |
| $(wildcard $(notdir $(ERLANG_MK_FILENAME))) \ |
| $(sort $(call core_find,include/,*.hrl)) \ |
| $(wildcard LICENSE*) \ |
| $(wildcard Makefile) \ |
| $(wildcard plugins.mk) \ |
| $(sort $(call core_find,priv/,*)) \ |
| $(wildcard README*) \ |
| $(wildcard rebar.config) \ |
| $(sort $(call core_find,src/,*)) |
| |
| HEX_TARBALL_OUTPUT_FILE ?= $(ERLANG_MK_TMP)/$(PROJECT).tar |
| |
| # @todo Need to check for rebar.config and/or the absence of DEPS to know |
| # whether a project will work with Rebar. |
| # |
| # @todo contributors licenses links in HEX_TARBALL_EXTRA_METADATA |
| |
| # In order to build the requirements metadata we look into DEPS. |
| # We do not require that the project use Hex dependencies, however |
| # Hex.pm does require that the package name and version numbers |
| # correspond to a real Hex package. |
| define hex_tarball_create.erl |
| Files0 = [$(call comma_list,$(patsubst %,"%",$(HEX_TARBALL_FILES)))], |
| Requirements0 = #{ |
| $(foreach d,$(DEPS), |
| <<"$(if $(subst hex,,$(call query_fetch_method,$d)),$d,$(if $(word 3,$(dep_$d)),$(word 3,$(dep_$d)),$d))">> => #{ |
| <<"app">> => <<"$d">>, |
| <<"optional">> => false, |
| <<"requirement">> => <<"$(call query_version,$d)">> |
| },) |
| $(if $(DEPS),dummy => dummy) |
| }, |
| Requirements = maps:remove(dummy, Requirements0), |
| Metadata0 = #{ |
| app => <<"$(strip $(PROJECT))">>, |
| build_tools => [<<"make">>, <<"rebar3">>], |
| description => <<"$(strip $(PROJECT_DESCRIPTION))">>, |
| files => [unicode:characters_to_binary(F) || F <- Files0], |
| name => <<"$(strip $(PROJECT))">>, |
| requirements => Requirements, |
| version => <<"$(strip $(PROJECT_VERSION))">> |
| }, |
| Metadata = Metadata0$(HEX_TARBALL_EXTRA_METADATA), |
| Files = [case file:read_file(F) of |
| {ok, Bin} -> |
| {F, Bin}; |
| {error, Reason} -> |
| io:format("Error trying to open file ~0p: ~0p~n", [F, Reason]), |
| halt(82) |
| end || F <- Files0], |
| case hex_tarball:create(Metadata, Files) of |
| {ok, #{tarball := Tarball}} -> |
| ok = file:write_file("$(strip $(HEX_TARBALL_OUTPUT_FILE))", Tarball), |
| halt(0); |
| {error, Reason} -> |
| io:format("Error ~0p~n", [Reason]), |
| halt(83) |
| end |
| endef |
| |
| hex_tar_verbose_0 = @echo " TAR $(notdir $(ERLANG_MK_TMP))/$(@F)"; |
| hex_tar_verbose_2 = set -x; |
| hex_tar_verbose = $(hex_tar_verbose_$(V)) |
| |
| $(HEX_TARBALL_OUTPUT_FILE): hex-core app |
| $(hex_tar_verbose) $(call erlang,$(call hex_tarball_create.erl)) |
| |
| hex-tarball-create: $(HEX_TARBALL_OUTPUT_FILE) |
| |
| define hex_release_publish_summary.erl |
| {ok, Tarball} = erl_tar:open("$(strip $(HEX_TARBALL_OUTPUT_FILE))", [read]), |
| ok = erl_tar:extract(Tarball, [{cwd, "$(ERLANG_MK_TMP)"}, {files, ["metadata.config"]}]), |
| {ok, Metadata} = file:consult("$(ERLANG_MK_TMP)/metadata.config"), |
| #{ |
| <<"name">> := Name, |
| <<"version">> := Version, |
| <<"files">> := Files, |
| <<"requirements">> := Deps |
| } = maps:from_list(Metadata), |
| io:format("Publishing ~s ~s~n Dependencies:~n", [Name, Version]), |
| case Deps of |
| [] -> |
| io:format(" (none)~n"); |
| _ -> |
| [begin |
| #{<<"app">> := DA, <<"requirement">> := DR} = maps:from_list(D), |
| io:format(" ~s ~s~n", [DA, DR]) |
| end || {_, D} <- Deps] |
| end, |
| io:format(" Included files:~n"), |
| [io:format(" ~s~n", [F]) || F <- Files], |
| io:format("You may also review the contents of the tarball file.~n" |
| "Please enter your secret key to proceed.~n"), |
| halt(0) |
| endef |
| |
| define hex_release_publish.erl |
| {ok, _} = application:ensure_all_started(ssl), |
| {ok, _} = application:ensure_all_started(inets), |
| Config = $(hex_config.erl), |
| ConfigF = Config#{api_key => <<"$(strip $1)">>}, |
| {ok, Tarball} = file:read_file("$(strip $(HEX_TARBALL_OUTPUT_FILE))"), |
| case hex_api_release:publish(ConfigF, Tarball, [{replace, $2}]) of |
| {ok, {200, _, #{}}} -> |
| io:format("Release replaced~n"), |
| halt(0); |
| {ok, {201, _, #{}}} -> |
| io:format("Release published~n"), |
| halt(0); |
| {ok, {Status, _, Errors}} -> |
| io:format("Error ~b: ~0p~n", [Status, Errors]), |
| halt(84) |
| end |
| endef |
| |
| hex-release-tarball: hex-core $(HEX_TARBALL_OUTPUT_FILE) |
| $(verbose) $(call erlang,$(call hex_release_publish_summary.erl)) |
| |
| hex-release-publish: hex-core hex-release-tarball |
| $(if $(HEX_SECRET),,$(eval HEX_SECRET := $(shell stty -echo; read -p "Secret: " secret; stty echo; echo $$secret) $(info ))) |
| $(gen_verbose) $(call erlang,$(call hex_release_publish.erl,$(HEX_SECRET),false)) |
| |
| hex-release-replace: hex-core hex-release-tarball |
| $(if $(HEX_SECRET),,$(eval HEX_SECRET := $(shell stty -echo; read -p "Secret: " secret; stty echo; echo $$secret) $(info ))) |
| $(gen_verbose) $(call erlang,$(call hex_release_publish.erl,$(HEX_SECRET),true)) |
| |
| define hex_release_delete.erl |
| {ok, _} = application:ensure_all_started(ssl), |
| {ok, _} = application:ensure_all_started(inets), |
| Config = $(hex_config.erl), |
| ConfigF = Config#{api_key => <<"$(strip $1)">>}, |
| case hex_api_release:delete(ConfigF, <<"$(strip $(PROJECT))">>, <<"$(strip $(PROJECT_VERSION))">>) of |
| {ok, {204, _, _}} -> |
| io:format("Release $(strip $(PROJECT_VERSION)) deleted~n"), |
| halt(0); |
| {ok, {Status, _, Errors}} -> |
| io:format("Error ~b: ~0p~n", [Status, Errors]), |
| halt(85) |
| end |
| endef |
| |
| hex-release-delete: hex-core |
| $(if $(HEX_SECRET),,$(eval HEX_SECRET := $(shell stty -echo; read -p "Secret: " secret; stty echo; echo $$secret) $(info ))) |
| $(gen_verbose) $(call erlang,$(call hex_release_delete.erl,$(HEX_SECRET))) |
| |
| define hex_release_retire.erl |
| {ok, _} = application:ensure_all_started(ssl), |
| {ok, _} = application:ensure_all_started(inets), |
| Config = $(hex_config.erl), |
| ConfigF = Config#{api_key => <<"$(strip $1)">>}, |
| Params = #{<<"reason">> => <<"$(strip $3)">>, <<"message">> => <<"$(strip $4)">>}, |
| case hex_api_release:retire(ConfigF, <<"$(strip $(PROJECT))">>, <<"$(strip $2)">>, Params) of |
| {ok, {204, _, _}} -> |
| io:format("Release $(strip $2) has been retired~n"), |
| halt(0); |
| {ok, {Status, _, Errors}} -> |
| io:format("Error ~b: ~0p~n", [Status, Errors]), |
| halt(86) |
| end |
| endef |
| |
| hex-release-retire: hex-core |
| $(if $(HEX_SECRET),,$(eval HEX_SECRET := $(shell stty -echo; read -p "Secret: " secret; stty echo; echo $$secret) $(info ))) |
| $(gen_verbose) $(call erlang,$(call hex_release_retire.erl,$(HEX_SECRET),\ |
| $(if $(HEX_VERSION),$(HEX_VERSION),$(PROJECT_VERSION)),\ |
| $(if $(HEX_REASON),$(HEX_REASON),invalid),\ |
| $(HEX_MESSAGE))) |
| |
| define hex_release_unretire.erl |
| {ok, _} = application:ensure_all_started(ssl), |
| {ok, _} = application:ensure_all_started(inets), |
| Config = $(hex_config.erl), |
| ConfigF = Config#{api_key => <<"$(strip $1)">>}, |
| case hex_api_release:unretire(ConfigF, <<"$(strip $(PROJECT))">>, <<"$(strip $2)">>) of |
| {ok, {204, _, _}} -> |
| io:format("Release $(strip $2) is not retired anymore~n"), |
| halt(0); |
| {ok, {Status, _, Errors}} -> |
| io:format("Error ~b: ~0p~n", [Status, Errors]), |
| halt(87) |
| end |
| endef |
| |
| hex-release-unretire: hex-core |
| $(if $(HEX_SECRET),,$(eval HEX_SECRET := $(shell stty -echo; read -p "Secret: " secret; stty echo; echo $$secret) $(info ))) |
| $(gen_verbose) $(call erlang,$(call hex_release_unretire.erl,$(HEX_SECRET),\ |
| $(if $(HEX_VERSION),$(HEX_VERSION),$(PROJECT_VERSION)))) |
| |
| HEX_DOCS_DOC_DIR ?= doc/ |
| HEX_DOCS_TARBALL_FILES ?= $(sort $(call core_find,$(HEX_DOCS_DOC_DIR),*)) |
| HEX_DOCS_TARBALL_OUTPUT_FILE ?= $(ERLANG_MK_TMP)/$(PROJECT)-docs.tar.gz |
| |
| $(HEX_DOCS_TARBALL_OUTPUT_FILE): hex-core app docs |
| $(hex_tar_verbose) tar czf $(HEX_DOCS_TARBALL_OUTPUT_FILE) -C $(HEX_DOCS_DOC_DIR) \ |
| $(HEX_DOCS_TARBALL_FILES:$(HEX_DOCS_DOC_DIR)%=%) |
| |
| hex-docs-tarball-create: $(HEX_DOCS_TARBALL_OUTPUT_FILE) |
| |
| define hex_docs_publish.erl |
| {ok, _} = application:ensure_all_started(ssl), |
| {ok, _} = application:ensure_all_started(inets), |
| Config = $(hex_config.erl), |
| ConfigF = Config#{api_key => <<"$(strip $1)">>}, |
| {ok, Tarball} = file:read_file("$(strip $(HEX_DOCS_TARBALL_OUTPUT_FILE))"), |
| case hex_api:post(ConfigF, |
| ["packages", "$(strip $(PROJECT))", "releases", "$(strip $(PROJECT_VERSION))", "docs"], |
| {"application/octet-stream", Tarball}) of |
| {ok, {Status, _, _}} when Status >= 200, Status < 300 -> |
| io:format("Docs published~n"), |
| halt(0); |
| {ok, {Status, _, Errors}} -> |
| io:format("Error ~b: ~0p~n", [Status, Errors]), |
| halt(88) |
| end |
| endef |
| |
| hex-docs-publish: hex-core hex-docs-tarball-create |
| $(if $(HEX_SECRET),,$(eval HEX_SECRET := $(shell stty -echo; read -p "Secret: " secret; stty echo; echo $$secret) $(info ))) |
| $(gen_verbose) $(call erlang,$(call hex_docs_publish.erl,$(HEX_SECRET))) |
| |
| define hex_docs_delete.erl |
| {ok, _} = application:ensure_all_started(ssl), |
| {ok, _} = application:ensure_all_started(inets), |
| Config = $(hex_config.erl), |
| ConfigF = Config#{api_key => <<"$(strip $1)">>}, |
| case hex_api:delete(ConfigF, |
| ["packages", "$(strip $(PROJECT))", "releases", "$(strip $2)", "docs"]) of |
| {ok, {Status, _, _}} when Status >= 200, Status < 300 -> |
| io:format("Docs removed~n"), |
| halt(0); |
| {ok, {Status, _, Errors}} -> |
| io:format("Error ~b: ~0p~n", [Status, Errors]), |
| halt(89) |
| end |
| endef |
| |
| hex-docs-delete: hex-core |
| $(if $(HEX_SECRET),,$(eval HEX_SECRET := $(shell stty -echo; read -p "Secret: " secret; stty echo; echo $$secret) $(info ))) |
| $(gen_verbose) $(call erlang,$(call hex_docs_delete.erl,$(HEX_SECRET),\ |
| $(if $(HEX_VERSION),$(HEX_VERSION),$(PROJECT_VERSION)))) |
| |
| # Copyright (c) 2015-2017, Loïc Hoguin <essen@ninenines.eu> |
| # This file is part of erlang.mk and subject to the terms of the ISC License. |
| |
| ifeq ($(filter proper,$(DEPS) $(TEST_DEPS)),proper) |
| .PHONY: proper |
| |
| # Targets. |
| |
| tests:: proper |
| |
| define proper_check.erl |
| $(call cover.erl) |
| code:add_pathsa([ |
| "$(call core_native_path,$(CURDIR)/ebin)", |
| "$(call core_native_path,$(DEPS_DIR)/*/ebin)", |
| "$(call core_native_path,$(TEST_DIR))"]), |
| Module = fun(M) -> |
| [true] =:= lists:usort([ |
| case atom_to_list(F) of |
| "prop_" ++ _ -> |
| io:format("Testing ~p:~p/0~n", [M, F]), |
| proper:quickcheck(M:F(), nocolors); |
| _ -> |
| true |
| end |
| || {F, 0} <- M:module_info(exports)]) |
| end, |
| try begin |
| CoverSetup(), |
| Res = case $(1) of |
| all -> [true] =:= lists:usort([Module(M) || M <- [$(call comma_list,$(3))]]); |
| module -> Module($(2)); |
| function -> proper:quickcheck($(2), nocolors) |
| end, |
| CoverExport("$(COVER_DATA_DIR)/proper.coverdata"), |
| Res |
| end of |
| true -> halt(0); |
| _ -> halt(1) |
| catch error:undef$(if $V,:Stacktrace) -> |
| io:format("Undefined property or module?~n$(if $V,~p~n)", [$(if $V,Stacktrace)]), |
| halt(0) |
| end. |
| endef |
| |
| ifdef t |
| ifeq (,$(findstring :,$(t))) |
| proper: test-build cover-data-dir |
| $(verbose) $(call erlang,$(call proper_check.erl,module,$(t))) |
| else |
| proper: test-build cover-data-dir |
| $(verbose) echo Testing $(t)/0 |
| $(verbose) $(call erlang,$(call proper_check.erl,function,$(t)())) |
| endif |
| else |
| proper: test-build cover-data-dir |
| $(eval MODULES := $(patsubst %,'%',$(sort $(notdir $(basename \ |
| $(wildcard ebin/*.beam) $(call core_find,$(TEST_DIR)/,*.beam)))))) |
| $(gen_verbose) $(call erlang,$(call proper_check.erl,all,undefined,$(MODULES))) |
| endif |
| endif |
| |
| # Copyright (c) 2015-2016, Loïc Hoguin <essen@ninenines.eu> |
| # This file is part of erlang.mk and subject to the terms of the ISC License. |
| |
| # Verbosity. |
| |
| proto_verbose_0 = @echo " PROTO " $(filter %.proto,$(?F)); |
| proto_verbose = $(proto_verbose_$(V)) |
| |
| # Core targets. |
| |
| ifneq ($(wildcard src/),) |
| ifneq ($(filter gpb protobuffs,$(BUILD_DEPS) $(DEPS)),) |
| PROTO_FILES := $(filter %.proto,$(ALL_SRC_FILES)) |
| ERL_FILES += $(addprefix src/,$(patsubst %.proto,%_pb.erl,$(notdir $(PROTO_FILES)))) |
| |
| ifeq ($(PROTO_FILES),) |
| $(ERLANG_MK_TMP)/last-makefile-change-protobuffs: |
| $(verbose) : |
| else |
| # Rebuild proto files when the Makefile changes. |
| # We exclude $(PROJECT).d to avoid a circular dependency. |
| $(ERLANG_MK_TMP)/last-makefile-change-protobuffs: $(filter-out $(PROJECT).d,$(MAKEFILE_LIST)) | $(ERLANG_MK_TMP) |
| $(verbose) if test -f $@; then \ |
| touch $(PROTO_FILES); \ |
| fi |
| $(verbose) touch $@ |
| |
| $(PROJECT).d:: $(ERLANG_MK_TMP)/last-makefile-change-protobuffs |
| endif |
| |
| ifeq ($(filter gpb,$(BUILD_DEPS) $(DEPS)),) |
| define compile_proto.erl |
| [begin |
| protobuffs_compile:generate_source(F, [ |
| {output_include_dir, "./include"}, |
| {output_src_dir, "./src"}]) |
| end || F <- string:tokens("$1", " ")], |
| halt(). |
| endef |
| else |
| define compile_proto.erl |
| [begin |
| gpb_compile:file(F, [ |
| $(foreach i,$(sort $(dir $(PROTO_FILES))),{i$(comma) "$i"}$(comma)) |
| {include_as_lib, true}, |
| {module_name_suffix, "_pb"}, |
| {o_hrl, "./include"}, |
| {o_erl, "./src"}, |
| {use_packages, true} |
| ]) |
| end || F <- string:tokens("$1", " ")], |
| halt(). |
| endef |
| endif |
| |
| ifneq ($(PROTO_FILES),) |
| $(PROJECT).d:: $(PROTO_FILES) |
| $(verbose) mkdir -p ebin/ include/ |
| $(if $(strip $?),$(proto_verbose) $(call erlang,$(call compile_proto.erl,$?))) |
| endif |
| endif |
| endif |
| |
| # Copyright (c) 2013-2016, Loïc Hoguin <essen@ninenines.eu> |
| # This file is part of erlang.mk and subject to the terms of the ISC License. |
| |
| ifeq ($(filter relx,$(BUILD_DEPS) $(DEPS) $(REL_DEPS)),relx) |
| .PHONY: relx-rel relx-relup distclean-relx-rel run |
| |
| # Configuration. |
| |
| RELX_CONFIG ?= $(CURDIR)/relx.config |
| RELX_CONFIG_SCRIPT ?= $(CURDIR)/relx.config.script |
| |
| RELX_OUTPUT_DIR ?= _rel |
| RELX_REL_EXT ?= |
| RELX_TAR ?= 1 |
| |
| ifdef SFX |
| RELX_TAR = 1 |
| endif |
| |
| # Core targets. |
| |
| ifeq ($(IS_DEP),) |
| ifneq ($(wildcard $(RELX_CONFIG))$(wildcard $(RELX_CONFIG_SCRIPT)),) |
| rel:: relx-rel |
| |
| relup:: relx-relup |
| endif |
| endif |
| |
| distclean:: distclean-relx-rel |
| |
| # Plugin-specific targets. |
| |
| define relx_get_config.erl |
| (fun() -> |
| Config0 = |
| case file:consult("$(call core_native_path,$(RELX_CONFIG))") of |
| {ok, Terms} -> |
| Terms; |
| {error, _} -> |
| [] |
| end, |
| case filelib:is_file("$(call core_native_path,$(RELX_CONFIG_SCRIPT))") of |
| true -> |
| Bindings = erl_eval:add_binding('CONFIG', Config0, erl_eval:new_bindings()), |
| {ok, Config1} = file:script("$(call core_native_path,$(RELX_CONFIG_SCRIPT))", Bindings), |
| Config1; |
| false -> |
| Config0 |
| end |
| end)() |
| endef |
| |
| define relx_release.erl |
| Config = $(call relx_get_config.erl), |
| {release, {Name, Vsn0}, _} = lists:keyfind(release, 1, Config), |
| Vsn = case Vsn0 of |
| {cmd, Cmd} -> os:cmd(Cmd); |
| semver -> ""; |
| {semver, _} -> ""; |
| {git, short} -> string:trim(os:cmd("git rev-parse --short HEAD"), both, "\n"); |
| {git, long} -> string:trim(os:cmd("git rev-parse HEAD"), both, "\n"); |
| VsnStr -> Vsn0 |
| end, |
| {ok, _} = relx:build_release(#{name => Name, vsn => Vsn}, Config ++ [{output_dir, "$(RELX_OUTPUT_DIR)"}]), |
| halt(0). |
| endef |
| |
| define relx_tar.erl |
| Config = $(call relx_get_config.erl), |
| {release, {Name, Vsn0}, _} = lists:keyfind(release, 1, Config), |
| Vsn = case Vsn0 of |
| {cmd, Cmd} -> os:cmd(Cmd); |
| semver -> ""; |
| {semver, _} -> ""; |
| {git, short} -> string:trim(os:cmd("git rev-parse --short HEAD"), both, "\n"); |
| {git, long} -> string:trim(os:cmd("git rev-parse HEAD"), both, "\n"); |
| VsnStr -> Vsn0 |
| end, |
| {ok, _} = relx:build_tar(#{name => Name, vsn => Vsn}, Config ++ [{output_dir, "$(RELX_OUTPUT_DIR)"}]), |
| halt(0). |
| endef |
| |
| define relx_relup.erl |
| Config = $(call relx_get_config.erl), |
| {release, {Name, Vsn0}, _} = lists:keyfind(release, 1, Config), |
| Vsn = case Vsn0 of |
| {cmd, Cmd} -> os:cmd(Cmd); |
| semver -> ""; |
| {semver, _} -> ""; |
| {git, short} -> string:trim(os:cmd("git rev-parse --short HEAD"), both, "\n"); |
| {git, long} -> string:trim(os:cmd("git rev-parse HEAD"), both, "\n"); |
| VsnStr -> Vsn0 |
| end, |
| {ok, _} = relx:build_relup(Name, Vsn, undefined, Config ++ [{output_dir, "$(RELX_OUTPUT_DIR)"}]), |
| halt(0). |
| endef |
| |
| relx-rel: rel-deps app |
| $(call erlang,$(call relx_release.erl),-pa ebin/) |
| $(verbose) $(MAKE) relx-post-rel |
| ifeq ($(RELX_TAR),1) |
| $(call erlang,$(call relx_tar.erl),-pa ebin/) |
| endif |
| |
| relx-relup: rel-deps app |
| $(call erlang,$(call relx_release.erl),-pa ebin/) |
| $(MAKE) relx-post-rel |
| $(call erlang,$(call relx_relup.erl),-pa ebin/) |
| ifeq ($(RELX_TAR),1) |
| $(call erlang,$(call relx_tar.erl),-pa ebin/) |
| endif |
| |
| distclean-relx-rel: |
| $(gen_verbose) rm -rf $(RELX_OUTPUT_DIR) |
| |
| # Default hooks. |
| relx-post-rel:: |
| $(verbose) : |
| |
| # Run target. |
| |
| ifeq ($(wildcard $(RELX_CONFIG))$(wildcard $(RELX_CONFIG_SCRIPT)),) |
| run:: |
| else |
| |
| define get_relx_release.erl |
| Config = $(call relx_get_config.erl), |
| {release, {Name, Vsn0}, _} = lists:keyfind(release, 1, Config), |
| Vsn = case Vsn0 of |
| {cmd, Cmd} -> os:cmd(Cmd); |
| semver -> ""; |
| {semver, _} -> ""; |
| {git, short} -> string:trim(os:cmd("git rev-parse --short HEAD"), both, "\n"); |
| {git, long} -> string:trim(os:cmd("git rev-parse HEAD"), both, "\n"); |
| VsnStr -> Vsn0 |
| end, |
| Extended = case lists:keyfind(extended_start_script, 1, Config) of |
| {_, true} -> "1"; |
| _ -> "" |
| end, |
| io:format("~s ~s ~s", [Name, Vsn, Extended]), |
| halt(0). |
| endef |
| |
| RELX_REL := $(shell $(call erlang,$(get_relx_release.erl))) |
| RELX_REL_NAME := $(word 1,$(RELX_REL)) |
| RELX_REL_VSN := $(word 2,$(RELX_REL)) |
| RELX_REL_CMD := $(if $(word 3,$(RELX_REL)),console) |
| |
| ifeq ($(PLATFORM),msys2) |
| RELX_REL_EXT := .cmd |
| endif |
| |
| run:: all |
| $(verbose) $(RELX_OUTPUT_DIR)/$(RELX_REL_NAME)/bin/$(RELX_REL_NAME)$(RELX_REL_EXT) $(RELX_REL_CMD) |
| |
| ifdef RELOAD |
| rel:: |
| $(verbose) $(RELX_OUTPUT_DIR)/$(RELX_REL_NAME)/bin/$(RELX_REL_NAME)$(RELX_REL_EXT) ping |
| $(verbose) $(RELX_OUTPUT_DIR)/$(RELX_REL_NAME)/bin/$(RELX_REL_NAME)$(RELX_REL_EXT) \ |
| eval "io:format(\"~p~n\", [c:lm()])." |
| endif |
| |
| help:: |
| $(verbose) printf "%s\n" "" \ |
| "Relx targets:" \ |
| " run Compile the project, build the release and run it" |
| |
| endif |
| endif |
| |
| # Copyright (c) 2015-2016, Loïc Hoguin <essen@ninenines.eu> |
| # Copyright (c) 2014, M Robert Martin <rob@version2beta.com> |
| # This file is contributed to erlang.mk and subject to the terms of the ISC License. |
| |
| .PHONY: shell |
| |
| # Configuration. |
| |
| SHELL_ERL ?= erl |
| SHELL_PATHS ?= $(CURDIR)/ebin $(TEST_DIR) |
| SHELL_OPTS ?= |
| |
| ALL_SHELL_DEPS_DIRS = $(addprefix $(DEPS_DIR)/,$(SHELL_DEPS)) |
| |
| # Core targets |
| |
| help:: |
| $(verbose) printf "%s\n" "" \ |
| "Shell targets:" \ |
| " shell Run an erlang shell with SHELL_OPTS or reasonable default" |
| |
| # Plugin-specific targets. |
| |
| $(foreach dep,$(SHELL_DEPS),$(eval $(call dep_target,$(dep)))) |
| |
| ifneq ($(SKIP_DEPS),) |
| build-shell-deps: |
| else |
| build-shell-deps: $(ALL_SHELL_DEPS_DIRS) |
| $(verbose) set -e; for dep in $(ALL_SHELL_DEPS_DIRS) ; do \ |
| if [ -z "$(strip $(FULL))" ] && [ ! -L $$dep ] && [ -f $$dep/ebin/dep_built ]; then \ |
| :; \ |
| else \ |
| $(MAKE) -C $$dep IS_DEP=1; \ |
| if [ ! -L $$dep ] && [ -d $$dep/ebin ]; then touch $$dep/ebin/dep_built; fi; \ |
| fi \ |
| done |
| endif |
| |
| shell:: build-shell-deps |
| $(gen_verbose) $(SHELL_ERL) -pa $(SHELL_PATHS) $(SHELL_OPTS) |
| |
| # Copyright 2017, Stanislaw Klekot <dozzie@jarowit.net> |
| # This file is part of erlang.mk and subject to the terms of the ISC License. |
| |
| .PHONY: distclean-sphinx sphinx |
| |
| # Configuration. |
| |
| SPHINX_BUILD ?= sphinx-build |
| SPHINX_SOURCE ?= doc |
| SPHINX_CONFDIR ?= |
| SPHINX_FORMATS ?= html |
| SPHINX_DOCTREES ?= $(ERLANG_MK_TMP)/sphinx.doctrees |
| SPHINX_OPTS ?= |
| |
| #sphinx_html_opts = |
| #sphinx_html_output = html |
| #sphinx_man_opts = |
| #sphinx_man_output = man |
| #sphinx_latex_opts = |
| #sphinx_latex_output = latex |
| |
| # Helpers. |
| |
| sphinx_build_0 = @echo " SPHINX" $1; $(SPHINX_BUILD) -N -q |
| sphinx_build_1 = $(SPHINX_BUILD) -N |
| sphinx_build_2 = set -x; $(SPHINX_BUILD) |
| sphinx_build = $(sphinx_build_$(V)) |
| |
| define sphinx.build |
| $(call sphinx_build,$1) -b $1 -d $(SPHINX_DOCTREES) $(if $(SPHINX_CONFDIR),-c $(SPHINX_CONFDIR)) $(SPHINX_OPTS) $(sphinx_$1_opts) -- $(SPHINX_SOURCE) $(call sphinx.output,$1) |
| |
| endef |
| |
| define sphinx.output |
| $(if $(sphinx_$1_output),$(sphinx_$1_output),$1) |
| endef |
| |
| # Targets. |
| |
| ifneq ($(wildcard $(if $(SPHINX_CONFDIR),$(SPHINX_CONFDIR),$(SPHINX_SOURCE))/conf.py),) |
| docs:: sphinx |
| distclean:: distclean-sphinx |
| endif |
| |
| help:: |
| $(verbose) printf "%s\n" "" \ |
| "Sphinx targets:" \ |
| " sphinx Generate Sphinx documentation." \ |
| "" \ |
| "ReST sources and 'conf.py' file are expected in directory pointed by" \ |
| "SPHINX_SOURCE ('doc' by default). SPHINX_FORMATS lists formats to build (only" \ |
| "'html' format is generated by default); target directory can be specified by" \ |
| 'setting sphinx_$${format}_output, for example: sphinx_html_output = output/html' \ |
| "Additional Sphinx options can be set in SPHINX_OPTS." |
| |
| # Plugin-specific targets. |
| |
| sphinx: |
| $(foreach F,$(SPHINX_FORMATS),$(call sphinx.build,$F)) |
| |
| distclean-sphinx: |
| $(gen_verbose) rm -rf $(filter-out $(SPHINX_SOURCE),$(foreach F,$(SPHINX_FORMATS),$(call sphinx.output,$F))) |
| |
| # Copyright (c) 2017, Jean-Sébastien Pédron <jean-sebastien@rabbitmq.com> |
| # This file is contributed to erlang.mk and subject to the terms of the ISC License. |
| |
| .PHONY: show-ERL_LIBS show-ERLC_OPTS show-TEST_ERLC_OPTS |
| |
| show-ERL_LIBS: |
| @echo $(ERL_LIBS) |
| |
| show-ERLC_OPTS: |
| @$(foreach opt,$(ERLC_OPTS) -pa ebin -I include,echo "$(opt)";) |
| |
| show-TEST_ERLC_OPTS: |
| @$(foreach opt,$(TEST_ERLC_OPTS) -pa ebin -I include,echo "$(opt)";) |
| |
| # Copyright (c) 2015-2016, Loïc Hoguin <essen@ninenines.eu> |
| # This file is part of erlang.mk and subject to the terms of the ISC License. |
| |
| ifeq ($(filter triq,$(DEPS) $(TEST_DEPS)),triq) |
| .PHONY: triq |
| |
| # Targets. |
| |
| tests:: triq |
| |
| define triq_check.erl |
| $(call cover.erl) |
| code:add_pathsa([ |
| "$(call core_native_path,$(CURDIR)/ebin)", |
| "$(call core_native_path,$(DEPS_DIR)/*/ebin)", |
| "$(call core_native_path,$(TEST_DIR))"]), |
| try begin |
| CoverSetup(), |
| Res = case $(1) of |
| all -> [true] =:= lists:usort([triq:check(M) || M <- [$(call comma_list,$(3))]]); |
| module -> triq:check($(2)); |
| function -> triq:check($(2)) |
| end, |
| CoverExport("$(COVER_DATA_DIR)/triq.coverdata"), |
| Res |
| end of |
| true -> halt(0); |
| _ -> halt(1) |
| catch error:undef$(if $V,:Stacktrace) -> |
| io:format("Undefined property or module?~n$(if $V,~p~n)", [$(if $V,Stacktrace)]), |
| halt(0) |
| end. |
| endef |
| |
| ifdef t |
| ifeq (,$(findstring :,$(t))) |
| triq: test-build cover-data-dir |
| $(verbose) $(call erlang,$(call triq_check.erl,module,$(t))) |
| else |
| triq: test-build cover-data-dir |
| $(verbose) echo Testing $(t)/0 |
| $(verbose) $(call erlang,$(call triq_check.erl,function,$(t)())) |
| endif |
| else |
| triq: test-build cover-data-dir |
| $(eval MODULES := $(patsubst %,'%',$(sort $(notdir $(basename \ |
| $(wildcard ebin/*.beam) $(call core_find,$(TEST_DIR)/,*.beam)))))) |
| $(gen_verbose) $(call erlang,$(call triq_check.erl,all,undefined,$(MODULES))) |
| endif |
| endif |
| |
| # Copyright (c) 2022, Loïc Hoguin <essen@ninenines.eu> |
| # This file is part of erlang.mk and subject to the terms of the ISC License. |
| |
| .PHONY: xref |
| |
| # Configuration. |
| |
| # We do not use locals_not_used or deprecated_function_calls |
| # because the compiler will error out by default in those |
| # cases with Erlang.mk. Deprecated functions may make sense |
| # in some cases but few libraries define them. We do not |
| # use exports_not_used by default because it hinders more |
| # than it helps library projects such as Cowboy. Finally, |
| # undefined_functions provides little that undefined_function_calls |
| # doesn't already provide, so it's not enabled by default. |
| XREF_CHECKS ?= [undefined_function_calls] |
| |
| # Instead of predefined checks a query can be evaluated |
| # using the Xref DSL. The $q variable is used in that case. |
| |
| # The scope is a list of keywords that correspond to |
| # application directories, being essentially an easy way |
| # to configure which applications to analyze. With: |
| # |
| # - app: . |
| # - apps: $(ALL_APPS_DIRS) |
| # - deps: $(ALL_DEPS_DIRS) |
| # - otp: Built-in Erlang/OTP applications. |
| # |
| # The default is conservative (app) and will not be |
| # appropriate for all types of queries (for example |
| # application_call requires adding all applications |
| # that might be called or they will not be found). |
| XREF_SCOPE ?= app # apps deps otp |
| |
| # If the above is not enough, additional application |
| # directories can be configured. |
| XREF_EXTRA_APP_DIRS ?= |
| |
| # As well as additional non-application directories. |
| XREF_EXTRA_DIRS ?= |
| |
| # Erlang.mk supports -ignore_xref([...]) with forms |
| # {M, F, A} | {F, A} | M, the latter ignoring whole |
| # modules. Ignores can also be provided project-wide. |
| XREF_IGNORE ?= [] |
| |
| # All callbacks may be ignored. Erlang.mk will ignore |
| # them automatically for exports_not_used (unless it |
| # is explicitly disabled by the user). |
| XREF_IGNORE_CALLBACKS ?= |
| |
| # Core targets. |
| |
| help:: |
| $(verbose) printf '%s\n' '' \ |
| 'Xref targets:' \ |
| ' xref Analyze the project using Xref' \ |
| ' xref q=QUERY Evaluate an Xref query' |
| |
| # Plugin-specific targets. |
| |
| define xref.erl |
| {ok, Xref} = xref:start([]), |
| Scope = [$(call comma_list,$(XREF_SCOPE))], |
| AppDirs0 = [$(call comma_list,$(foreach d,$(XREF_EXTRA_APP_DIRS),"$d"))], |
| AppDirs1 = case lists:member(otp, Scope) of |
| false -> AppDirs0; |
| true -> |
| RootDir = code:root_dir(), |
| AppDirs0 ++ [filename:dirname(P) || P <- code:get_path(), lists:prefix(RootDir, P)] |
| end, |
| AppDirs2 = case lists:member(deps, Scope) of |
| false -> AppDirs1; |
| true -> [$(call comma_list,$(foreach d,$(ALL_DEPS_DIRS),"$d"))] ++ AppDirs1 |
| end, |
| AppDirs3 = case lists:member(apps, Scope) of |
| false -> AppDirs2; |
| true -> [$(call comma_list,$(foreach d,$(ALL_APPS_DIRS),"$d"))] ++ AppDirs2 |
| end, |
| AppDirs = case lists:member(app, Scope) of |
| false -> AppDirs3; |
| true -> ["../$(notdir $(CURDIR))"|AppDirs3] |
| end, |
| [{ok, _} = xref:add_application(Xref, AppDir, [{builtins, true}]) || AppDir <- AppDirs], |
| ExtraDirs = [$(call comma_list,$(foreach d,$(XREF_EXTRA_DIRS),"$d"))], |
| [{ok, _} = xref:add_directory(Xref, ExtraDir, [{builtins, true}]) || ExtraDir <- ExtraDirs], |
| ok = xref:set_library_path(Xref, code:get_path() -- (["ebin", "."] ++ AppDirs ++ ExtraDirs)), |
| Checks = case {$1, is_list($2)} of |
| {check, true} -> $2; |
| {check, false} -> [$2]; |
| {query, _} -> [$2] |
| end, |
| FinalRes = [begin |
| IsInformational = case $1 of |
| query -> true; |
| check -> |
| is_tuple(Check) andalso |
| lists:member(element(1, Check), |
| [call, use, module_call, module_use, application_call, application_use]) |
| end, |
| {ok, Res0} = case $1 of |
| check -> xref:analyze(Xref, Check); |
| query -> xref:q(Xref, Check) |
| end, |
| Res = case IsInformational of |
| true -> Res0; |
| false -> |
| lists:filter(fun(R) -> |
| {Mod, InMFA, MFA} = case R of |
| {InMFA0 = {M, _, _}, MFA0} -> {M, InMFA0, MFA0}; |
| {M, _, _} -> {M, R, R} |
| end, |
| Attrs = try |
| Mod:module_info(attributes) |
| catch error:undef -> |
| [] |
| end, |
| InlineIgnores = lists:flatten([ |
| [case V of |
| M when is_atom(M) -> {M, '_', '_'}; |
| {F, A} -> {Mod, F, A}; |
| _ -> V |
| end || V <- Values] |
| || {ignore_xref, Values} <- Attrs]), |
| BuiltinIgnores = [ |
| {eunit_test, wrapper_test_exported_, 0} |
| ], |
| DoCallbackIgnores = case {Check, "$(strip $(XREF_IGNORE_CALLBACKS))"} of |
| {exports_not_used, ""} -> true; |
| {_, "0"} -> false; |
| _ -> true |
| end, |
| CallbackIgnores = case DoCallbackIgnores of |
| false -> []; |
| true -> |
| Behaviors = lists:flatten([ |
| [BL || {behavior, BL} <- Attrs], |
| [BL || {behaviour, BL} <- Attrs] |
| ]), |
| [{Mod, CF, CA} || B <- Behaviors, {CF, CA} <- B:behaviour_info(callbacks)] |
| end, |
| WideIgnores = if |
| is_list($(XREF_IGNORE)) -> |
| [if is_atom(I) -> {I, '_', '_'}; true -> I end |
| || I <- $(XREF_IGNORE)]; |
| true -> [$(XREF_IGNORE)] |
| end, |
| Ignores = InlineIgnores ++ BuiltinIgnores ++ CallbackIgnores ++ WideIgnores, |
| not (lists:member(InMFA, Ignores) |
| orelse lists:member(MFA, Ignores) |
| orelse lists:member({Mod, '_', '_'}, Ignores)) |
| end, Res0) |
| end, |
| case Res of |
| [] -> ok; |
| _ when IsInformational -> |
| case Check of |
| {call, {CM, CF, CA}} -> |
| io:format("Functions that ~s:~s/~b calls:~n", [CM, CF, CA]); |
| {use, {CM, CF, CA}} -> |
| io:format("Function ~s:~s/~b is called by:~n", [CM, CF, CA]); |
| {module_call, CMod} -> |
| io:format("Modules that ~s calls:~n", [CMod]); |
| {module_use, CMod} -> |
| io:format("Module ~s is used by:~n", [CMod]); |
| {application_call, CApp} -> |
| io:format("Applications that ~s calls:~n", [CApp]); |
| {application_use, CApp} -> |
| io:format("Application ~s is used by:~n", [CApp]); |
| _ when $1 =:= query -> |
| io:format("Query ~s returned:~n", [Check]) |
| end, |
| [case R of |
| {{InM, InF, InA}, {M, F, A}} -> |
| io:format("- ~s:~s/~b called by ~s:~s/~b~n", |
| [M, F, A, InM, InF, InA]); |
| {M, F, A} -> |
| io:format("- ~s:~s/~b~n", [M, F, A]); |
| ModOrApp -> |
| io:format("- ~s~n", [ModOrApp]) |
| end || R <- Res], |
| ok; |
| _ -> |
| [case {Check, R} of |
| {undefined_function_calls, {{InM, InF, InA}, {M, F, A}}} -> |
| io:format("Undefined function ~s:~s/~b called by ~s:~s/~b~n", |
| [M, F, A, InM, InF, InA]); |
| {undefined_functions, {M, F, A}} -> |
| io:format("Undefined function ~s:~s/~b~n", [M, F, A]); |
| {locals_not_used, {M, F, A}} -> |
| io:format("Unused local function ~s:~s/~b~n", [M, F, A]); |
| {exports_not_used, {M, F, A}} -> |
| io:format("Unused exported function ~s:~s/~b~n", [M, F, A]); |
| {deprecated_function_calls, {{InM, InF, InA}, {M, F, A}}} -> |
| io:format("Deprecated function ~s:~s/~b called by ~s:~s/~b~n", |
| [M, F, A, InM, InF, InA]); |
| {deprecated_functions, {M, F, A}} -> |
| io:format("Deprecated function ~s:~s/~b~n", [M, F, A]); |
| _ -> |
| io:format("~p: ~p~n", [Check, R]) |
| end || R <- Res], |
| error |
| end |
| end || Check <- Checks], |
| stopped = xref:stop(Xref), |
| case lists:usort(FinalRes) of |
| [ok] -> halt(0); |
| _ -> halt(1) |
| end |
| endef |
| |
| xref: deps app |
| ifdef q |
| $(verbose) $(call erlang,$(call xref.erl,query,"$q"),-pa ebin/) |
| else |
| $(verbose) $(call erlang,$(call xref.erl,check,$(XREF_CHECKS)),-pa ebin/) |
| endif |
| |
| # Copyright (c) 2016, Loïc Hoguin <essen@ninenines.eu> |
| # Copyright (c) 2015, Viktor Söderqvist <viktor@zuiderkwast.se> |
| # This file is part of erlang.mk and subject to the terms of the ISC License. |
| |
| COVER_REPORT_DIR ?= cover |
| COVER_DATA_DIR ?= $(COVER_REPORT_DIR) |
| |
| ifdef COVER |
| COVER_APPS ?= $(notdir $(ALL_APPS_DIRS)) |
| COVER_DEPS ?= |
| COVER_EXCLUDE_MODS ?= |
| endif |
| |
| # Code coverage for Common Test. |
| |
| ifdef COVER |
| ifdef CT_RUN |
| ifneq ($(wildcard $(TEST_DIR)),) |
| test-build:: $(TEST_DIR)/ct.cover.spec |
| |
| $(TEST_DIR)/ct.cover.spec: cover-data-dir |
| $(gen_verbose) printf "%s\n" \ |
| "{incl_app, '$(PROJECT)', details}." \ |
| "{incl_dirs, '$(PROJECT)', [\"$(call core_native_path,$(CURDIR)/ebin)\" \ |
| $(foreach a,$(COVER_APPS),$(comma) \"$(call core_native_path,$(APPS_DIR)/$a/ebin)\") \ |
| $(foreach d,$(COVER_DEPS),$(comma) \"$(call core_native_path,$(DEPS_DIR)/$d/ebin)\")]}." \ |
| '{export,"$(call core_native_path,$(abspath $(COVER_DATA_DIR))/ct.coverdata)"}.' \ |
| "{excl_mods, '$(PROJECT)', [$(call comma_list,$(COVER_EXCLUDE_MODS))]}." > $@ |
| |
| CT_RUN += -cover $(TEST_DIR)/ct.cover.spec |
| endif |
| endif |
| endif |
| |
| # Code coverage for other tools. |
| |
| ifdef COVER |
| define cover.erl |
| CoverSetup = fun() -> |
| Dirs = ["$(call core_native_path,$(CURDIR)/ebin)" |
| $(foreach a,$(COVER_APPS),$(comma) "$(call core_native_path,$(APPS_DIR)/$a/ebin)") |
| $(foreach d,$(COVER_DEPS),$(comma) "$(call core_native_path,$(DEPS_DIR)/$d/ebin)")], |
| Excludes = [$(call comma_list,$(foreach e,$(COVER_EXCLUDE_MODS),"$e"))], |
| [case file:list_dir(Dir) of |
| {error, enotdir} -> false; |
| {error, _} -> halt(2); |
| {ok, Files} -> |
| BeamFiles = [filename:join(Dir, File) || |
| File <- Files, |
| not lists:member(filename:basename(File, ".beam"), Excludes), |
| filename:extension(File) =:= ".beam"], |
| case cover:compile_beam(BeamFiles) of |
| {error, _} -> halt(1); |
| _ -> true |
| end |
| end || Dir <- Dirs] |
| end, |
| CoverExport = fun(Filename) -> cover:export(Filename) end, |
| endef |
| else |
| define cover.erl |
| CoverSetup = fun() -> ok end, |
| CoverExport = fun(_) -> ok end, |
| endef |
| endif |
| |
| # Core targets |
| |
| ifdef COVER |
| ifneq ($(COVER_REPORT_DIR),) |
| tests:: |
| $(verbose) $(MAKE) --no-print-directory cover-report |
| endif |
| |
| cover-data-dir: | $(COVER_DATA_DIR) |
| |
| $(COVER_DATA_DIR): |
| $(verbose) mkdir -p $(COVER_DATA_DIR) |
| else |
| cover-data-dir: |
| endif |
| |
| clean:: coverdata-clean |
| |
| ifneq ($(COVER_REPORT_DIR),) |
| distclean:: cover-report-clean |
| endif |
| |
| help:: |
| $(verbose) printf "%s\n" "" \ |
| "Cover targets:" \ |
| " cover-report Generate a HTML coverage report from previously collected" \ |
| " cover data." \ |
| " all.coverdata Merge all coverdata files into all.coverdata." \ |
| "" \ |
| "If COVER=1 is set, coverage data is generated by the targets eunit and ct. The" \ |
| "target tests additionally generates a HTML coverage report from the combined" \ |
| "coverdata files from each of these testing tools. HTML reports can be disabled" \ |
| "by setting COVER_REPORT_DIR to empty." |
| |
| # Plugin specific targets |
| |
| COVERDATA = $(filter-out $(COVER_DATA_DIR)/all.coverdata,$(wildcard $(COVER_DATA_DIR)/*.coverdata)) |
| |
| .PHONY: coverdata-clean |
| coverdata-clean: |
| $(gen_verbose) rm -f $(COVER_DATA_DIR)/*.coverdata $(TEST_DIR)/ct.cover.spec |
| |
| # Merge all coverdata files into one. |
| define cover_export.erl |
| $(foreach f,$(COVERDATA),cover:import("$(f)") == ok orelse halt(1),) |
| cover:export("$(COVER_DATA_DIR)/$@"), halt(0). |
| endef |
| |
| all.coverdata: $(COVERDATA) cover-data-dir |
| $(gen_verbose) $(call erlang,$(cover_export.erl)) |
| |
| # These are only defined if COVER_REPORT_DIR is non-empty. Set COVER_REPORT_DIR to |
| # empty if you want the coverdata files but not the HTML report. |
| ifneq ($(COVER_REPORT_DIR),) |
| |
| .PHONY: cover-report-clean cover-report |
| |
| cover-report-clean: |
| $(gen_verbose) rm -rf $(COVER_REPORT_DIR) |
| ifneq ($(COVER_REPORT_DIR),$(COVER_DATA_DIR)) |
| $(if $(shell ls -A $(COVER_DATA_DIR)/),,$(verbose) rmdir $(COVER_DATA_DIR)) |
| endif |
| |
| ifeq ($(COVERDATA),) |
| cover-report: |
| else |
| |
| # Modules which include eunit.hrl always contain one line without coverage |
| # because eunit defines test/0 which is never called. We compensate for this. |
| EUNIT_HRL_MODS = $(subst $(space),$(comma),$(shell \ |
| grep -H -e '^\s*-include.*include/eunit\.hrl"' src/*.erl \ |
| | sed "s/^src\/\(.*\)\.erl:.*/'\1'/" | uniq)) |
| |
| define cover_report.erl |
| $(foreach f,$(COVERDATA),cover:import("$(f)") == ok orelse halt(1),) |
| Ms = cover:imported_modules(), |
| [cover:analyse_to_file(M, "$(COVER_REPORT_DIR)/" ++ atom_to_list(M) |
| ++ ".COVER.html", [html]) || M <- Ms], |
| Report = [begin {ok, R} = cover:analyse(M, module), R end || M <- Ms], |
| EunitHrlMods = [$(EUNIT_HRL_MODS)], |
| Report1 = [{M, {Y, case lists:member(M, EunitHrlMods) of |
| true -> N - 1; false -> N end}} || {M, {Y, N}} <- Report], |
| TotalY = lists:sum([Y || {_, {Y, _}} <- Report1]), |
| TotalN = lists:sum([N || {_, {_, N}} <- Report1]), |
| Perc = fun(Y, N) -> case Y + N of 0 -> 100; S -> round(100 * Y / S) end end, |
| TotalPerc = Perc(TotalY, TotalN), |
| {ok, F} = file:open("$(COVER_REPORT_DIR)/index.html", [write]), |
| io:format(F, "<!DOCTYPE html><html>~n" |
| "<head><meta charset=\"UTF-8\">~n" |
| "<title>Coverage report</title></head>~n" |
| "<body>~n", []), |
| io:format(F, "<h1>Coverage</h1>~n<p>Total: ~p%</p>~n", [TotalPerc]), |
| io:format(F, "<table><tr><th>Module</th><th>Coverage</th></tr>~n", []), |
| [io:format(F, "<tr><td><a href=\"~p.COVER.html\">~p</a></td>" |
| "<td>~p%</td></tr>~n", |
| [M, M, Perc(Y, N)]) || {M, {Y, N}} <- Report1], |
| How = "$(subst $(space),$(comma)$(space),$(basename $(COVERDATA)))", |
| Date = "$(shell date -u "+%Y-%m-%dT%H:%M:%SZ")", |
| io:format(F, "</table>~n" |
| "<p>Generated using ~s and erlang.mk on ~s.</p>~n" |
| "</body></html>", [How, Date]), |
| halt(). |
| endef |
| |
| cover-report: |
| $(verbose) mkdir -p $(COVER_REPORT_DIR) |
| $(gen_verbose) $(call erlang,$(cover_report.erl)) |
| |
| endif |
| endif # ifneq ($(COVER_REPORT_DIR),) |
| |
| # Copyright (c) 2016, Loïc Hoguin <essen@ninenines.eu> |
| # This file is part of erlang.mk and subject to the terms of the ISC License. |
| |
| .PHONY: sfx |
| |
| ifdef RELX_REL |
| ifdef SFX |
| |
| # Configuration. |
| |
| SFX_ARCHIVE ?= $(RELX_OUTPUT_DIR)/$(RELX_REL_NAME)/$(RELX_REL_NAME)-$(RELX_REL_VSN).tar.gz |
| SFX_OUTPUT_FILE ?= $(RELX_OUTPUT_DIR)/$(RELX_REL_NAME).run |
| |
| # Core targets. |
| |
| rel:: sfx |
| |
| # Plugin-specific targets. |
| |
| define sfx_stub |
| #!/bin/sh |
| |
| TMPDIR=`mktemp -d` |
| ARCHIVE=`awk '/^__ARCHIVE_BELOW__$$/ {print NR + 1; exit 0;}' $$0` |
| FILENAME=$$(basename $$0) |
| REL=$${FILENAME%.*} |
| |
| tail -n+$$ARCHIVE $$0 | tar -xzf - -C $$TMPDIR |
| |
| $$TMPDIR/bin/$$REL console |
| RET=$$? |
| |
| rm -rf $$TMPDIR |
| |
| exit $$RET |
| |
| __ARCHIVE_BELOW__ |
| endef |
| |
| sfx: |
| $(verbose) $(call core_render,sfx_stub,$(SFX_OUTPUT_FILE)) |
| $(gen_verbose) cat $(SFX_ARCHIVE) >> $(SFX_OUTPUT_FILE) |
| $(verbose) chmod +x $(SFX_OUTPUT_FILE) |
| |
| endif |
| endif |
| |
| # Copyright (c) 2013-2017, Loïc Hoguin <essen@ninenines.eu> |
| # This file is part of erlang.mk and subject to the terms of the ISC License. |
| |
| # External plugins. |
| |
| DEP_PLUGINS ?= |
| |
| $(foreach p,$(DEP_PLUGINS),\ |
| $(eval $(if $(findstring /,$p),\ |
| $(call core_dep_plugin,$p,$(firstword $(subst /, ,$p))),\ |
| $(call core_dep_plugin,$p/plugins.mk,$p)))) |
| |
| help:: help-plugins |
| |
| help-plugins:: |
| $(verbose) : |
| |
| # Copyright (c) 2013-2015, Loïc Hoguin <essen@ninenines.eu> |
| # Copyright (c) 2015-2016, Jean-Sébastien Pédron <jean-sebastien@rabbitmq.com> |
| # This file is part of erlang.mk and subject to the terms of the ISC License. |
| |
| # Fetch dependencies recursively (without building them). |
| |
| .PHONY: fetch-deps fetch-doc-deps fetch-rel-deps fetch-test-deps \ |
| fetch-shell-deps |
| |
| .PHONY: $(ERLANG_MK_RECURSIVE_DEPS_LIST) \ |
| $(ERLANG_MK_RECURSIVE_DOC_DEPS_LIST) \ |
| $(ERLANG_MK_RECURSIVE_REL_DEPS_LIST) \ |
| $(ERLANG_MK_RECURSIVE_TEST_DEPS_LIST) \ |
| $(ERLANG_MK_RECURSIVE_SHELL_DEPS_LIST) |
| |
| fetch-deps: $(ERLANG_MK_RECURSIVE_DEPS_LIST) |
| fetch-doc-deps: $(ERLANG_MK_RECURSIVE_DOC_DEPS_LIST) |
| fetch-rel-deps: $(ERLANG_MK_RECURSIVE_REL_DEPS_LIST) |
| fetch-test-deps: $(ERLANG_MK_RECURSIVE_TEST_DEPS_LIST) |
| fetch-shell-deps: $(ERLANG_MK_RECURSIVE_SHELL_DEPS_LIST) |
| |
| ifneq ($(SKIP_DEPS),) |
| $(ERLANG_MK_RECURSIVE_DEPS_LIST) \ |
| $(ERLANG_MK_RECURSIVE_DOC_DEPS_LIST) \ |
| $(ERLANG_MK_RECURSIVE_REL_DEPS_LIST) \ |
| $(ERLANG_MK_RECURSIVE_TEST_DEPS_LIST) \ |
| $(ERLANG_MK_RECURSIVE_SHELL_DEPS_LIST): |
| $(verbose) :> $@ |
| else |
| # By default, we fetch "normal" dependencies. They are also included no |
| # matter the type of requested dependencies. |
| # |
| # $(ALL_DEPS_DIRS) includes $(BUILD_DEPS). |
| |
| $(ERLANG_MK_RECURSIVE_DEPS_LIST): $(LOCAL_DEPS_DIRS) $(ALL_DEPS_DIRS) |
| $(ERLANG_MK_RECURSIVE_DOC_DEPS_LIST): $(LOCAL_DEPS_DIRS) $(ALL_DEPS_DIRS) $(ALL_DOC_DEPS_DIRS) |
| $(ERLANG_MK_RECURSIVE_REL_DEPS_LIST): $(LOCAL_DEPS_DIRS) $(ALL_DEPS_DIRS) $(ALL_REL_DEPS_DIRS) |
| $(ERLANG_MK_RECURSIVE_TEST_DEPS_LIST): $(LOCAL_DEPS_DIRS) $(ALL_DEPS_DIRS) $(ALL_TEST_DEPS_DIRS) |
| $(ERLANG_MK_RECURSIVE_SHELL_DEPS_LIST): $(LOCAL_DEPS_DIRS) $(ALL_DEPS_DIRS) $(ALL_SHELL_DEPS_DIRS) |
| |
| # Allow to use fetch-deps and $(DEP_TYPES) to fetch multiple types of |
| # dependencies with a single target. |
| ifneq ($(filter doc,$(DEP_TYPES)),) |
| $(ERLANG_MK_RECURSIVE_DEPS_LIST): $(ALL_DOC_DEPS_DIRS) |
| endif |
| ifneq ($(filter rel,$(DEP_TYPES)),) |
| $(ERLANG_MK_RECURSIVE_DEPS_LIST): $(ALL_REL_DEPS_DIRS) |
| endif |
| ifneq ($(filter test,$(DEP_TYPES)),) |
| $(ERLANG_MK_RECURSIVE_DEPS_LIST): $(ALL_TEST_DEPS_DIRS) |
| endif |
| ifneq ($(filter shell,$(DEP_TYPES)),) |
| $(ERLANG_MK_RECURSIVE_DEPS_LIST): $(ALL_SHELL_DEPS_DIRS) |
| endif |
| |
| ERLANG_MK_RECURSIVE_TMP_LIST := $(abspath $(ERLANG_MK_TMP)/recursive-tmp-deps-$(shell echo $$PPID).log) |
| |
| $(ERLANG_MK_RECURSIVE_DEPS_LIST) \ |
| $(ERLANG_MK_RECURSIVE_DOC_DEPS_LIST) \ |
| $(ERLANG_MK_RECURSIVE_REL_DEPS_LIST) \ |
| $(ERLANG_MK_RECURSIVE_TEST_DEPS_LIST) \ |
| $(ERLANG_MK_RECURSIVE_SHELL_DEPS_LIST): | $(ERLANG_MK_TMP) |
| ifeq ($(IS_APP)$(IS_DEP),) |
| $(verbose) rm -f $(ERLANG_MK_RECURSIVE_TMP_LIST) |
| endif |
| $(verbose) touch $(ERLANG_MK_RECURSIVE_TMP_LIST) |
| $(verbose) set -e; for dep in $^ ; do \ |
| if ! grep -qs ^$$dep$$ $(ERLANG_MK_RECURSIVE_TMP_LIST); then \ |
| echo $$dep >> $(ERLANG_MK_RECURSIVE_TMP_LIST); \ |
| if grep -qs -E "^[[:blank:]]*include[[:blank:]]+(erlang\.mk|.*/erlang\.mk|.*ERLANG_MK_FILENAME.*)$$" \ |
| $$dep/GNUmakefile $$dep/makefile $$dep/Makefile; then \ |
| $(MAKE) -C $$dep fetch-deps \ |
| IS_DEP=1 \ |
| ERLANG_MK_RECURSIVE_TMP_LIST=$(ERLANG_MK_RECURSIVE_TMP_LIST); \ |
| fi \ |
| fi \ |
| done |
| ifeq ($(IS_APP)$(IS_DEP),) |
| $(verbose) sort < $(ERLANG_MK_RECURSIVE_TMP_LIST) | \ |
| uniq > $(ERLANG_MK_RECURSIVE_TMP_LIST).sorted |
| $(verbose) cmp -s $(ERLANG_MK_RECURSIVE_TMP_LIST).sorted $@ \ |
| || mv $(ERLANG_MK_RECURSIVE_TMP_LIST).sorted $@ |
| $(verbose) rm -f $(ERLANG_MK_RECURSIVE_TMP_LIST).sorted |
| $(verbose) rm $(ERLANG_MK_RECURSIVE_TMP_LIST) |
| endif |
| endif # ifneq ($(SKIP_DEPS),) |
| |
| # List dependencies recursively. |
| |
| .PHONY: list-deps list-doc-deps list-rel-deps list-test-deps \ |
| list-shell-deps |
| |
| list-deps: $(ERLANG_MK_RECURSIVE_DEPS_LIST) |
| list-doc-deps: $(ERLANG_MK_RECURSIVE_DOC_DEPS_LIST) |
| list-rel-deps: $(ERLANG_MK_RECURSIVE_REL_DEPS_LIST) |
| list-test-deps: $(ERLANG_MK_RECURSIVE_TEST_DEPS_LIST) |
| list-shell-deps: $(ERLANG_MK_RECURSIVE_SHELL_DEPS_LIST) |
| |
| list-deps list-doc-deps list-rel-deps list-test-deps list-shell-deps: |
| $(verbose) cat $^ |
| |
| # Query dependencies recursively. |
| |
| .PHONY: query-deps query-doc-deps query-rel-deps query-test-deps \ |
| query-shell-deps |
| |
| QUERY ?= name fetch_method repo version |
| |
| define query_target |
| $(1): $(2) clean-tmp-query.log |
| ifeq ($(IS_APP)$(IS_DEP),) |
| $(verbose) rm -f $(4) |
| endif |
| $(verbose) $(foreach dep,$(3),\ |
| echo $(PROJECT): $(foreach q,$(QUERY),$(call query_$(q),$(dep))) >> $(4) ;) |
| $(if $(filter-out query-deps,$(1)),,\ |
| $(verbose) set -e; for dep in $(3) ; do \ |
| if grep -qs ^$$$$dep$$$$ $(ERLANG_MK_TMP)/query.log; then \ |
| :; \ |
| else \ |
| echo $$$$dep >> $(ERLANG_MK_TMP)/query.log; \ |
| $(MAKE) -C $(DEPS_DIR)/$$$$dep $$@ QUERY="$(QUERY)" IS_DEP=1 || true; \ |
| fi \ |
| done) |
| ifeq ($(IS_APP)$(IS_DEP),) |
| $(verbose) touch $(4) |
| $(verbose) cat $(4) |
| endif |
| endef |
| |
| clean-tmp-query.log: |
| ifeq ($(IS_DEP),) |
| $(verbose) rm -f $(ERLANG_MK_TMP)/query.log |
| endif |
| |
| $(eval $(call query_target,query-deps,$(ERLANG_MK_RECURSIVE_DEPS_LIST),$(BUILD_DEPS) $(DEPS),$(ERLANG_MK_QUERY_DEPS_FILE))) |
| $(eval $(call query_target,query-doc-deps,$(ERLANG_MK_RECURSIVE_DOC_DEPS_LIST),$(DOC_DEPS),$(ERLANG_MK_QUERY_DOC_DEPS_FILE))) |
| $(eval $(call query_target,query-rel-deps,$(ERLANG_MK_RECURSIVE_REL_DEPS_LIST),$(REL_DEPS),$(ERLANG_MK_QUERY_REL_DEPS_FILE))) |
| $(eval $(call query_target,query-test-deps,$(ERLANG_MK_RECURSIVE_TEST_DEPS_LIST),$(TEST_DEPS),$(ERLANG_MK_QUERY_TEST_DEPS_FILE))) |
| $(eval $(call query_target,query-shell-deps,$(ERLANG_MK_RECURSIVE_SHELL_DEPS_LIST),$(SHELL_DEPS),$(ERLANG_MK_QUERY_SHELL_DEPS_FILE))) |