1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200
|
import pytest
from briefcase.commands import PackageCommand
from briefcase.commands.base import full_options
from briefcase.config import AppConfig
from ...utils import DummyConsole, create_file
class DefaultPackageCommand(PackageCommand):
# An instance of PackageCommand that inherits the default
# behavior of packaging format handling. Define an output_format
# to satisfy the requirements of that interface.
output_format = "default"
# Two methods that are required by the interface, but are not needed
# for these tests.
def binary_path(self, app):
return NotImplementedError()
def distribution_path(self, app):
return NotImplementedError()
@pytest.fixture
def default_package_command(tmp_path):
return DefaultPackageCommand(base_path=tmp_path, console=DummyConsole())
class DummyPackageCommand(PackageCommand):
"""A dummy package command that doesn't actually do anything.
It only serves to track which actions would be performed.
"""
# Platform and format contain upper case to test case normalization
platform = "Tester"
output_format = "Dummy"
description = "Dummy package command"
@property
def packaging_formats(self):
return ["pkg", "box"]
@property
def default_packaging_format(self):
return "pkg"
def __init__(self, *args, **kwargs):
kwargs.setdefault("console", DummyConsole())
super().__init__(*args, apps={}, **kwargs)
self.actions = []
def briefcase_toml(self, app):
# default any app to an empty `briefcase.toml`
return self._briefcase_toml.get(app, {})
def binary_path(self, app):
return self.bundle_path(app) / f"{app.app_name}.bin"
def distribution_path(self, app):
return self.dist_path / f"{app.app_name}-{app.version}.{app.packaging_format}"
def verify_host(self):
super().verify_host()
self.actions.append(("verify-host",))
def verify_tools(self):
super().verify_tools()
self.actions.append(("verify-tools",))
def finalize_app_config(self, app):
super().finalize_app_config(app=app)
self.actions.append(("finalize-app-config", app.app_name))
def verify_app_template(self, app):
super().verify_app_template(app=app)
self.actions.append(("verify-app-template", app.app_name))
def verify_app_tools(self, app):
super().verify_app_tools(app=app)
self.actions.append(("verify-app-tools", app.app_name))
def package_app(self, app, **kwargs):
self.actions.append(("package", app.app_name, kwargs.copy()))
return full_options({"package_state": app.app_name}, kwargs)
# These commands override the default behavior, simply tracking that
# they were invoked, rather than instantiating a Create/Update/Build command.
# This is for testing purposes.
def create_command(self, app, **kwargs):
self.actions.append(("create", app.app_name, kwargs.copy()))
# Remove arguments consumed by the underlying call to create_app()
kwargs.pop("test_mode", None)
return full_options({"create_state": app.app_name}, kwargs)
def update_command(self, app, **kwargs):
self.actions.append(("update", app.app_name, kwargs.copy()))
# Remove arguments consumed by the underlying call to update_app()
kwargs.pop("update", None)
kwargs.pop("update_requirements", None)
kwargs.pop("update_resources", None)
kwargs.pop("update_support", None)
kwargs.pop("test_mode", None)
return full_options({"update_state": app.app_name}, kwargs)
def build_command(self, app, **kwargs):
self.actions.append(("build", app.app_name, kwargs.copy()))
# Remove arguments consumed by the underlying call to build_app()
kwargs.pop("update", None)
kwargs.pop("update_requirements", None)
kwargs.pop("update_resources", None)
kwargs.pop("update_support", None)
kwargs.pop("test_mode", None)
return full_options({"build_state": app.app_name}, kwargs)
@pytest.fixture
def package_command(tmp_path):
(tmp_path / "base_path").mkdir()
return DummyPackageCommand(base_path=tmp_path / "base_path")
@pytest.fixture
def first_app_config():
return AppConfig(
app_name="first",
bundle="com.example",
version="0.0.1",
description="The first simple app",
sources=["src/first"],
license={"file": "LICENSE"},
)
@pytest.fixture
def first_app_unbuilt(first_app_config, tmp_path):
# The same fixture as first_app_config,
# but ensures that the bundle for the app exists
create_file(
tmp_path / "base_path/build/first/tester/dummy/first.dummy",
"first.dummy",
)
return first_app_config
@pytest.fixture
def first_app(first_app_unbuilt, tmp_path):
# The same fixture as first_app_uncompiled,
# but ensures that the binary for the app exists
create_file(
tmp_path / "base_path/build/first/tester/dummy/first.bin",
"first.bin",
)
return first_app_unbuilt
@pytest.fixture
def second_app_config():
return AppConfig(
app_name="second",
bundle="com.example",
version="0.0.2",
description="The second simple app",
sources=["src/second"],
license={"file": "LICENSE"},
)
@pytest.fixture
def second_app_uncompiled(second_app_config, tmp_path):
# The same fixture as second_app_config,
# but ensures that the bundle for the app exists
create_file(
tmp_path
/ "base_path"
/ "build"
/ "second"
/ "tester"
/ "dummy"
/ "second.dummy",
"second.dummy",
)
return second_app_config
@pytest.fixture
def second_app(second_app_uncompiled, tmp_path):
# The same fixture as second_app_uncompiled,
# but ensures that the binary for the app exists
create_file(
tmp_path / "base_path/build/second/tester/dummy/second.bin",
"second.bin",
)
return second_app_uncompiled
|