Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add support for custom packages #717

Merged
merged 15 commits into from
Feb 19, 2024
5 changes: 3 additions & 2 deletions aea/cli/fetch.py
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
# -*- coding: utf-8 -*-
# ------------------------------------------------------------------------------
#
# Copyright 2022-2023 Valory AG
# Copyright 2022-2024 Valory AG
# Copyright 2018-2021 Fetch.AI Limited
#
# Licensed under the Apache License, Version 2.0 (the "License");
Expand Down Expand Up @@ -45,6 +45,7 @@
AGENTS,
CONNECTION,
CONTRACT,
CUSTOM,
DEFAULT_AEA_CONFIG_FILE,
PROTOCOL,
SKILL,
Expand Down Expand Up @@ -266,7 +267,7 @@ def _fetch_agent_deps(ctx: Context) -> None:

:param ctx: context object.
"""
for item_type in (PROTOCOL, CONTRACT, CONNECTION, SKILL):
for item_type in (PROTOCOL, CONTRACT, CUSTOM, CONNECTION, SKILL):
item_type_plural = "{}s".format(item_type)
required_items = cast(set, getattr(ctx.agent_config, item_type_plural))
required_items_check = required_items.copy()
Expand Down
19 changes: 13 additions & 6 deletions aea/cli/scaffold.py
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
# -*- coding: utf-8 -*-
# ------------------------------------------------------------------------------
#
# Copyright 2021-2023 Valory AG
# Copyright 2021-2024 Valory AG
# Copyright 2018-2019 Fetch.AI Limited
#
# Licensed under the Apache License, Version 2.0 (the "License");
Expand Down Expand Up @@ -41,18 +41,17 @@
validate_package_name,
)
from aea.configurations.base import PackageType, PublicId
from aea.configurations.constants import ( # noqa: F401 # pylint: disable=unused-import
from aea.configurations.constants import (
BUILD,
CONNECTION,
CONTRACT,
CONTRACTS,
CUSTOM,
DEFAULT_AEA_CONFIG_FILE,
DEFAULT_CONNECTION_CONFIG_FILE,
DEFAULT_CONTRACT_CONFIG_FILE,
DEFAULT_PROTOCOL_CONFIG_FILE,
DEFAULT_SKILL_CONFIG_FILE,
DEFAULT_VERSION,
DOTTED_PATH_MODULE_ELEMENT_SEPARATOR,
PACKAGE_TYPE_TO_CONFIG_FILE,
PROTOCOL,
SCAFFOLD_PUBLIC_ID,
SKILL,
Expand Down Expand Up @@ -155,6 +154,14 @@ def contract(
add_contract_abi(ctx, contract_name, Path(contract_abi_path))


@scaffold.command()
@click.argument("name", type=str, required=True)
@pass_ctx
def custom(ctx: Context, name: str) -> None:
"""Scaffold a custom component package."""
scaffold_item(ctx, CUSTOM, name)


@scaffold.command()
@click.argument("protocol_name", type=str, required=True)
@click.option("-y", "--yes", is_flag=True, default=False)
Expand Down Expand Up @@ -205,7 +212,7 @@ def scaffold_item(ctx: Context, item_type: str, item_name: str) -> None:
validate_package_name(item_name)
author_name = ctx.agent_config.author
loader = getattr(ctx, f"{item_type}_loader")
default_config_filename = globals()[f"DEFAULT_{item_type.upper()}_CONFIG_FILE"]
default_config_filename = PACKAGE_TYPE_TO_CONFIG_FILE[item_type]

to_local_registry = ctx.config.get("to_local_registry")

Expand Down
10 changes: 9 additions & 1 deletion aea/cli/utils/context.py
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
# -*- coding: utf-8 -*-
# ------------------------------------------------------------------------------
#
# Copyright 2022-2023 Valory AG
# Copyright 2022-2024 Valory AG
# Copyright 2018-2021 Fetch.AI Limited
#
# Licensed under the Apache License, Version 2.0 (the "License");
Expand Down Expand Up @@ -155,6 +155,13 @@ def contract_loader(self) -> ConfigLoader:
PackageType.CONTRACT, skip_aea_validation=self.skip_aea_validation
)

@property
def custom_loader(self) -> ConfigLoader:
"""Get the custom loader."""
return ConfigLoader.from_configuration_type(
PackageType.CUSTOM, skip_aea_validation=self.skip_aea_validation
)

def set_config(self, key: str, value: Any) -> None:
"""
Set a config.
Expand Down Expand Up @@ -263,6 +270,7 @@ def _update_dependencies(updates: Dependencies) -> None:
for item_type in (
PackageType.PROTOCOL,
PackageType.CONTRACT,
PackageType.CUSTOM,
PackageType.CONNECTION,
PackageType.SKILL,
PackageType.AGENT,
Expand Down
128 changes: 121 additions & 7 deletions aea/configurations/base.py
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
# -*- coding: utf-8 -*-
# ------------------------------------------------------------------------------
#
# Copyright 2021-2023 Valory AG
# Copyright 2021-2024 Valory AG
# Copyright 2018-2019 Fetch.AI Limited
#
# Licensed under the Apache License, Version 2.0 (the "License");
Expand Down Expand Up @@ -48,9 +48,11 @@
from aea.configurations.constants import (
CONNECTIONS,
CONTRACTS,
CUSTOMS,
DEFAULT_AEA_CONFIG_FILE,
DEFAULT_CONNECTION_CONFIG_FILE,
DEFAULT_CONTRACT_CONFIG_FILE,
DEFAULT_CUSTOM_COMPONENT_CONFIG_FILE,
DEFAULT_FINGERPRINT_IGNORE_PATTERNS,
DEFAULT_LICENSE,
DEFAULT_LOGGING_CONFIG,
Expand Down Expand Up @@ -146,6 +148,8 @@ def _get_default_configuration_file_name_from_type(
return DEFAULT_CONTRACT_CONFIG_FILE
if item_type == PackageType.SERVICE:
return DEFAULT_SERVICE_CONFIG_FILE
if item_type == PackageType.CUSTOM:
return DEFAULT_CUSTOM_COMPONENT_CONFIG_FILE
raise ValueError( # pragma: no cover
"Item type not valid: {}".format(str(item_type))
)
Expand Down Expand Up @@ -964,6 +968,7 @@ class SkillConfig(ComponentConfiguration):
"connections",
"protocols",
"contracts",
"customs",
"skills",
"dependencies",
"description",
Expand All @@ -987,6 +992,7 @@ def __init__(
connections: Optional[Set[PublicId]] = None,
protocols: Optional[Set[PublicId]] = None,
contracts: Optional[Set[PublicId]] = None,
customs: Optional[Set[PublicId]] = None,
skills: Optional[Set[PublicId]] = None,
dependencies: Optional[Dependencies] = None,
description: str = "",
Expand All @@ -1008,6 +1014,7 @@ def __init__(
self.connections = connections if connections is not None else set()
self.protocols = protocols if protocols is not None else set()
self.contracts = contracts if contracts is not None else set()
self.customs = customs if customs is not None else set()
self.skills = skills if skills is not None else set()
self.dependencies = dependencies if dependencies is not None else {}
self.description = description
Expand Down Expand Up @@ -1040,6 +1047,12 @@ def package_dependencies(self) -> Set[ComponentId]:
for connection_id in self.connections
}
)
.union(
{
ComponentId(ComponentType.CUSTOM, custom_id)
for custom_id in self.customs
}
)
)

@property
Expand All @@ -1063,6 +1076,7 @@ def json(self) -> Dict:
"fingerprint_ignore_patterns": self.fingerprint_ignore_patterns,
CONNECTIONS: sorted(map(str, self.connections)),
CONTRACTS: sorted(map(str, self.contracts)),
CUSTOMS: sorted(map(str, self.customs)),
PROTOCOLS: sorted(map(str, self.protocols)),
SKILLS: sorted(map(str, self.skills)),
"behaviours": {key: b.json for key, b in self.behaviours.read_all()},
Expand Down Expand Up @@ -1097,6 +1111,7 @@ def _create_or_update_from_json(
connections = {PublicId.from_str(id_) for id_ in obj.get(CONNECTIONS, set())}
protocols = {PublicId.from_str(id_) for id_ in obj.get(PROTOCOLS, set())}
contracts = {PublicId.from_str(id_) for id_ in obj.get(CONTRACTS, set())}
customs = {PublicId.from_str(id_) for id_ in obj.get(CUSTOMS, set())}
skills = {PublicId.from_str(id_) for id_ in obj.get(SKILLS, set())}
dependencies = dependencies_from_json(obj.get("dependencies", {}))
description = cast(str, obj.get("description", ""))
Expand All @@ -1112,6 +1127,7 @@ def _create_or_update_from_json(
connections=connections,
protocols=protocols,
contracts=contracts,
customs=customs,
skills=skills,
dependencies=dependencies,
description=description,
Expand Down Expand Up @@ -1211,6 +1227,7 @@ class AgentConfig(PackageConfiguration):
"protocols",
"skills",
"contracts",
"customs",
"period",
"execution_timeout",
"max_reactions",
Expand Down Expand Up @@ -1294,6 +1311,7 @@ def __init__( # pylint: disable=too-many-arguments,too-many-locals
)
self.connections = set() # type: Set[PublicId]
self.contracts = set() # type: Set[PublicId]
self.customs = set() # type: Set[PublicId]
self.protocols = set() # type: Set[PublicId]
self.skills = set() # type: Set[PublicId]

Expand Down Expand Up @@ -1341,6 +1359,7 @@ def component_configurations(self, d: Dict[ComponentId, Dict]) -> None:
PackageType.PROTOCOL: {epid.without_hash() for epid in self.protocols},
PackageType.CONNECTION: {epid.without_hash() for epid in self.connections},
PackageType.CONTRACT: {epid.without_hash() for epid in self.contracts},
PackageType.CUSTOM: {epid.without_hash() for epid in self.customs},
PackageType.SKILL: {epid.without_hash() for epid in self.skills},
}
for component_id, component_configuration in d.items():
Expand Down Expand Up @@ -1368,13 +1387,14 @@ def package_dependencies(self) -> Set[ComponentId]:
skills = set(
ComponentId(ComponentType.SKILL, public_id) for public_id in self.skills
)

contracts = set(
ComponentId(ComponentType.CONTRACT, public_id)
for public_id in self.contracts
)

return set.union(protocols, contracts, connections, skills)
customs = set(
ComponentId(ComponentType.CUSTOM, public_id) for public_id in self.customs
)
return set.union(protocols, contracts, customs, connections, skills)

@property
def private_key_paths_dict(self) -> Dict[str, str]:
Expand Down Expand Up @@ -1421,9 +1441,12 @@ def json(self) -> Dict:
CONTRACTS: sorted(map(str, self.contracts)),
PROTOCOLS: sorted(map(str, self.protocols)),
SKILLS: sorted(map(str, self.skills)),
"default_connection": str(self.default_connection)
if self.default_connection is not None
else None,
CUSTOMS: sorted(map(str, self.customs)),
"default_connection": (
DavidMinarsch marked this conversation as resolved.
Show resolved Hide resolved
str(self.default_connection)
if self.default_connection is not None
else None
),
"default_ledger": self.default_ledger,
"required_ledgers": self.required_ledgers or [],
"default_routing": {
Expand Down Expand Up @@ -1540,6 +1563,14 @@ def _create_or_update_from_json(
)
)

# parse custom public ids
agent_config.customs = set(
map(
PublicId.from_str,
obj.get(CUSTOMS, []),
)
)

# parse protocol public ids
agent_config.protocols = set(
map(
Expand Down Expand Up @@ -1575,6 +1606,7 @@ def all_components_id(self) -> List[ComponentId]:
ComponentType.PROTOCOL: self.protocols,
ComponentType.CONNECTION: self.connections,
ComponentType.CONTRACT: self.contracts,
ComponentType.CUSTOM: self.customs,
ComponentType.SKILL: self.skills,
}
result = []
Expand Down Expand Up @@ -1879,6 +1911,87 @@ def package_dependencies(self) -> Set[ComponentId]:
}


class CustomComponentConfig(PackageConfiguration):
"""Custom component configuratiopn."""

default_configuration_filename = DEFAULT_CUSTOM_COMPONENT_CONFIG_FILE
package_type = PackageType.CUSTOM
component_type = ComponentType.CUSTOM
schema = "custom-config_schema.json"

FIELDS_ALLOWED_TO_UPDATE: FrozenSet[str] = frozenset(
["config", "cert_requests", "is_abstract", "build_directory"]
)

def __init__(
self,
name: SimpleIdOrStr,
author: SimpleIdOrStr,
version: str = "",
license_: str = "",
aea_version: str = "",
description: str = "",
fingerprint: Optional[Dict[str, str]] = None,
fingerprint_ignore_patterns: Optional[Sequence[str]] = None,
dependencies: Optional[Dependencies] = None,
**kwargs: Any,
) -> None:
"""Initialize a custom configuration object."""
super().__init__(
name,
author,
version,
license_,
aea_version,
fingerprint,
fingerprint_ignore_patterns,
)
self.dependencies = dependencies if dependencies is not None else {}
self.description = description
self.kwargs = kwargs

def get(self, name: str) -> Any:
"""Get parameter."""
return self.kwargs.get(name)

def set(self, name: str, value: Any) -> None:
"""Set extra parameter value."""
self.kwargs[name] = value

@property
def json(self) -> Dict:
"""Return the JSON representation."""
result = OrderedDict(
{
"name": self.name,
"author": self.author,
"version": self.version,
"type": self.component_type.value,
"description": self.description,
"license": self.license,
"aea_version": self.aea_version,
"fingerprint": self.fingerprint,
"fingerprint_ignore_patterns": self.fingerprint_ignore_patterns,
"dependencies": dependencies_to_json(self.dependencies),
**self.kwargs,
}
)
return result

@classmethod
def _create_or_update_from_json(
cls, obj: Dict, instance: Optional["CustomComponentConfig"] = None
) -> "CustomComponentConfig":
"""Initialize from a JSON object."""
params = {**(instance.json if instance else {}), **copy(obj)}
params["dependencies"] = cast(
Dependencies, dependencies_from_json(obj.get("dependencies", {}))
)
return cast(
CustomComponentConfig, cls._apply_params_to_instance(params, instance)
)


"""The following functions are called from aea.cli.utils."""


Expand Down Expand Up @@ -2045,4 +2158,5 @@ def _get_public_id_from_file(
PackageType.CONNECTION: ConnectionConfig,
PackageType.SKILL: SkillConfig,
PackageType.CONTRACT: ContractConfig,
PackageType.CUSTOM: CustomComponentConfig,
}
Loading
Loading