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

Plugging new parametrization inside instrumentation and optimization #391

Merged
merged 150 commits into from
Jan 17, 2020
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
Show all changes
150 commits
Select commit Hold shift + click to select a range
2c50068
Prepare a new instrumentation pattern
jrapin Nov 14, 2019
f12070d
Value as property
jrapin Nov 14, 2019
fc87a3e
setter
jrapin Nov 14, 2019
da3e7b7
Random state and constraint
jrapin Nov 15, 2019
c3ab658
naming
jrapin Nov 15, 2019
41a86b4
nits
jrapin Nov 15, 2019
7f0f9a7
Split core features and secondary features
jrapin Nov 15, 2019
fce691b
More explicit structure
jrapin Nov 16, 2019
a27e0c1
missing
jrapin Nov 16, 2019
1c822e9
Feature test
jrapin Nov 16, 2019
1c7c523
Feature test
jrapin Nov 16, 2019
38020a6
fix
jrapin Nov 16, 2019
9c10389
Add name setter
jrapin Nov 16, 2019
1292002
Compliant dict
jrapin Nov 16, 2019
c185d7e
simplified spawn
jrapin Nov 16, 2019
4cbf118
Solve random_state propagation
jrapin Nov 16, 2019
20c315e
Solve random_state propagation
jrapin Nov 16, 2019
9ce1cd6
Constraint support
jrapin Nov 16, 2019
402c2f4
Array recombination
jrapin Nov 16, 2019
7b10af6
recombine
jrapin Nov 16, 2019
233dfc5
ParametersList
jrapin Nov 16, 2019
9cbfece
Starting choice parameter
jrapin Nov 16, 2019
9760530
Renaming mel-syd
jrapin Dec 1, 2019
ef9920e
Add generation + renaming (han-par)
jrapin Dec 7, 2019
6c8c55f
Added preliminary choice test
jrapin Dec 8, 2019
b9748d0
Better test
jrapin Dec 8, 2019
22d8b4a
Simplified access
jrapin Dec 9, 2019
ac71642
More simplification
jrapin Dec 9, 2019
900c025
Better access
jrapin Dec 9, 2019
3e33192
Start solving Choice test
jrapin Dec 9, 2019
ea74e47
Fixing all Choice main tests
jrapin Dec 9, 2019
6959a6a
Merge branch 'master' into instrumentation_v3
jrapin Dec 10, 2019
c4da7b9
nits
jrapin Dec 10, 2019
eafd9ce
Add instrumentation class
jrapin Dec 10, 2019
6ebc342
Restructure
jrapin Dec 10, 2019
b00178a
Merge branch 'master' into instrumentation_v3
jrapin Dec 10, 2019
7ad71a7
Simplify spawn
jrapin Dec 10, 2019
834582f
nits
jrapin Dec 10, 2019
91b08c6
Basic implementation of scalar
jrapin Dec 10, 2019
159e79d
Merge branch 'master' into instrumentation_v3
jrapin Dec 11, 2019
a01539d
Adding exponent to array
jrapin Dec 11, 2019
87c7c63
Log distributed data
jrapin Dec 11, 2019
56c7c69
Warning for bounds
jrapin Dec 11, 2019
b4ee7e1
Typing fix
jrapin Dec 11, 2019
5e78ac0
Better bound warning
jrapin Dec 11, 2019
9798394
Default to mutable sigma
jrapin Dec 11, 2019
bc3ea96
Better yet
jrapin Dec 11, 2019
e91c8f7
Samping
jrapin Dec 11, 2019
b11a59d
nit
jrapin Dec 11, 2019
4c01ca3
mypyfix
jrapin Dec 12, 2019
a2b6593
Convert other instances to std
jrapin Dec 12, 2019
9ba6758
Correction
jrapin Dec 12, 2019
4f9608e
Better tested conversion
jrapin Dec 12, 2019
51c0f00
Warning against lambdas
jrapin Dec 12, 2019
83ef082
Simplified initialization
jrapin Dec 12, 2019
a8d2511
Remove Ng prefix for simplicitys sake
jrapin Dec 12, 2019
98278f1
Disallow logarithmic with non-positive value
jrapin Dec 12, 2019
3ee9e5f
Activate other bounds, yet to be tested
jrapin Dec 12, 2019
ceacae3
Tested constraints
jrapin Dec 12, 2019
fe32abb
Removed warning
jrapin Dec 12, 2019
610effa
typing
jrapin Dec 12, 2019
813f6f3
Restructure as parametrization
jrapin Dec 13, 2019
9381212
Remove useless ignore
jrapin Dec 13, 2019
7868f87
Cast to int + beginning docstrings
jrapin Dec 13, 2019
c59d61a
Hopefully working typing
jrapin Dec 13, 2019
e16265a
Non-deterministic by default
jrapin Dec 13, 2019
77c99ec
Better docstrings
jrapin Dec 13, 2019
5aae810
More docstrings again
jrapin Dec 13, 2019
b8a9b33
Docstrings...
jrapin Dec 13, 2019
f473e54
Updated naming pattern and tested
jrapin Dec 13, 2019
c1717d9
Initial tag system
jrapin Dec 13, 2019
a559b6f
Updates and simplifications
jrapin Dec 16, 2019
83eaf5e
Add OrderedChoice
jrapin Dec 16, 2019
6040625
Share code for selection parameters
jrapin Dec 16, 2019
13e6a67
Names
jrapin Dec 16, 2019
d9a8ec7
Better docstrings
jrapin Dec 16, 2019
68c260f
Small comment
jrapin Dec 16, 2019
5f306be
file renaming
jrapin Dec 16, 2019
502b8d9
Compatibility for OrderedDiscrete
jrapin Dec 16, 2019
537efaf
[PR on instrumentation_v3] Convert all subparameters into Parameter (…
jrapin Dec 18, 2019
c98e6c1
Choice length
jrapin Dec 18, 2019
7125b77
Make Variable an Parameter/Instrumentation
jrapin Dec 19, 2019
b1a4f33
More tests
jrapin Dec 19, 2019
d3b8e2d
Constant correction
jrapin Dec 19, 2019
a2bce6f
Typing
jrapin Dec 19, 2019
5b3b0c0
Change typing to Parameter
jrapin Dec 19, 2019
28c60a3
Test both pipelines
jrapin Dec 19, 2019
3925292
Update core.py
jrapin Dec 19, 2019
d235a6f
Typing
jrapin Dec 19, 2019
2cf7566
details
jrapin Dec 19, 2019
ad839d7
Add set cheap constraint for compatiblity
jrapin Dec 19, 2019
d492182
skip
jrapin Dec 19, 2019
a2187f4
nit
jrapin Dec 19, 2019
484be3d
Merge base parameter and parameter
jrapin Dec 19, 2019
bc7ae1f
comment and renaming
jrapin Dec 19, 2019
2ba4ee6
comments
jrapin Dec 19, 2019
c1a82c1
Update description of constraint + start fixing sampling
jrapin Dec 20, 2019
0d7e108
Simplify choice
jrapin Dec 20, 2019
b9dea96
Move discretization and transforms modules
jrapin Dec 20, 2019
f8b6b8b
isinstance
jrapin Dec 20, 2019
f2e4017
Trying to remove some old code
jrapin Dec 20, 2019
1cd1756
Sanity check
jrapin Dec 20, 2019
94b2ac9
Merge branch 'instrumentation_v3' into variable_as_instrumentation
jrapin Dec 20, 2019
731ca05
Remove all Instrumentation
jrapin Dec 20, 2019
8d13ea7
Plug new instrumentation
jrapin Dec 20, 2019
f95a828
Improve perf with new Array
jrapin Dec 20, 2019
3c70b19
Flag failing instrumentations
jrapin Dec 20, 2019
a71db0d
Add a freezing mechanism
jrapin Dec 21, 2019
72080b8
Merging parametrization with freezing
jrapin Dec 21, 2019
bf0db82
Freezing and faster compatibility
jrapin Dec 21, 2019
546d51d
Start updating descriptros
jrapin Dec 21, 2019
038ec54
Updated descriptor pattern
jrapin Dec 21, 2019
8cd3eac
Enforce keyword arg
jrapin Dec 24, 2019
b6bdbb9
Solve initial value issue
jrapin Dec 24, 2019
56e308f
Update benchmark
jrapin Dec 24, 2019
1f2b9fc
Update functions
jrapin Dec 24, 2019
987cb1a
Use new choces
jrapin Dec 24, 2019
15076cb
transfor descriptors to children
jrapin Dec 24, 2019
b85b26b
Solve nan issue
jrapin Dec 24, 2019
4657833
No name shortening
jrapin Dec 24, 2019
2426f43
Merge branch 'master' into instrumentation_v3
jrapin Dec 24, 2019
d6eb182
Start merging master
jrapin Dec 24, 2019
4281962
Cleaner typing
jrapin Dec 25, 2019
01e6b9d
Add helpers
jrapin Dec 25, 2019
7bdecac
Merging master
jrapin Dec 25, 2019
0890214
Merge branch 'master' into instrumentation_v3
jrapin Dec 25, 2019
521cc56
Merge branch 'instrumentation_v3' into variable_as_instrumentation
jrapin Dec 25, 2019
3557470
Basic update of the documentation
jrapin Dec 25, 2019
0b9dbdd
Start adding deprecation warnings
jrapin Dec 25, 2019
4256956
Merge master
jrapin Dec 26, 2019
e558760
Merging master
jrapin Dec 26, 2019
160d3dc
Removing old constraint calls
jrapin Dec 27, 2019
64eea33
Remove from_value in favor of spawn_child param
jrapin Dec 27, 2019
f726de6
Deprecation warning for set_cheap_constraint_checker
jrapin Dec 27, 2019
1dab9a0
Merge branch 'instrumentation_v3' into variable_as_instrumentation
jrapin Dec 27, 2019
988ec11
Deprecation warnings for old descriptors
jrapin Dec 27, 2019
bac0150
Merging master (containing parameter)
jrapin Dec 30, 2019
c88893f
Merge master with ExperimentFunction (#432)
jrapin Jan 6, 2020
5281af4
Merge master again
jrapin Jan 6, 2020
efdf342
Minor documentation edit
jrapin Jan 6, 2020
4cd6de9
Change typing import to tp
jrapin Jan 6, 2020
6742296
Merge master
jrapin Jan 6, 2020
13ffb23
Solve compatibility bug
jrapin Jan 7, 2020
955942b
Merging master
jrapin Jan 7, 2020
7134287
Merge branch 'master' into variable_as_instrumentation
jrapin Jan 14, 2020
43d2014
merge
jrapin Jan 15, 2020
06b7c63
Merge branch 'master' into variable_as_instrumentation
jrapin Jan 15, 2020
0a17db6
Center and reduce standardized_space relatively to reference (#461)
jrapin Jan 16, 2020
675d929
Add test for offset
jrapin Jan 17, 2020
8791fb8
Merge branch 'master' into variable_as_instrumentation
jrapin Jan 17, 2020
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Prev Previous commit
Next Next commit
Cast to int + beginning docstrings
  • Loading branch information
jrapin committed Dec 13, 2019
commit 7868f8704ca07ffdb7dcbac1695044d9ab27bddb
51 changes: 46 additions & 5 deletions nevergrad/parametrization/core.py
Original file line number Diff line number Diff line change
Expand Up @@ -57,6 +57,9 @@ def _get_parameter_value(self, name: str) -> t.Any:

@property
def dimension(self) -> int:
"""Dimension of the standardized space for this parameter
i.e size of the vector returned by get_std_data()
"""
if self._dimension is None:
try:
self._dimension = self.get_std_data().size
Expand All @@ -65,17 +68,30 @@ def dimension(self) -> int:
return self._dimension

def mutate(self) -> None:
"""Mutate subparameters of the instance, and then its value
"""
self.subparameters.mutate()
data = self.get_std_data() # pylint: disable=assignment-from-no-return
# let's assume the random state is already there (next class)
self.set_std_data(data + self.random_state.normal(size=data.shape)) # type: ignore

def sample(self: BP) -> BP:
"""Sample a new instance of the parameter.
This usually means spawning a child and mutating it.
"""
child = self.spawn_child()
child.mutate()
return child

def recombine(self: BP, *others: BP) -> None:
"""Update value and subparameters of this instance by combining it with
other instances.

Parameters
----------
*others: Parameter
other instances of the same type than this instance.
"""
raise NotSupportedError(f"Recombination is not implemented for {self.name}") # type: ignore

def get_std_data(self: BP, instance: t.Optional[BP] = None) -> np.ndarray:
Expand Down Expand Up @@ -116,11 +132,13 @@ def __init__(self, **subparameters: t.Any) -> None:

@property
def generation(self) -> int:
"""generation of the parameter
"""generation of the parameter (children are current generation + 1)
"""
return self._generation

def get_value_hash(self) -> t.Hashable:
"""Hashable object representing the current value of the instance
"""
val = self.value
if isinstance(val, (str, bytes, float, int)):
return val
Expand All @@ -130,6 +148,14 @@ def get_value_hash(self) -> t.Hashable:
raise NotSupportedError(f"Value hash is not supported for object {self.name}")

def get_data_hash(self) -> t.Hashable:
"""Hashable object representing the current standardized data of the object.

Note
----
- this differs from the value hash, since the value is sometimes randomly sampled from the data
- standardized data does not account for the full state of the instance (it does not contain
data from subparameters)
"""
return self.get_std_data().tobytes()

def _get_name(self) -> str:
Expand Down Expand Up @@ -161,13 +187,28 @@ def set_name(self: P, name: str) -> P:

# %% Constraint management

def complies_with_constraint(self) -> bool:
def satisfies_constraint(self) -> bool:
"""Whether the instance complies with the constraints added through
the "register_cheap_constraint" method
"""
if not self._constraint_checkers:
return True
val = self.value
return all(func(val) for func in self._constraint_checkers)

def register_cheap_constraint(self, func: t.Callable[[t.Any], bool]) -> None:
"""Registers a new constraint on the parameter values.

Parameter
---------
func: Callable
function which, given the value of the instance, returns whether it satisfies the constraint.

Note
- this is only for checking after mutation/recombination/etc if the value still satisfy the constraints.
The constraint is not used in those processes.
- constraints should be fast to compute.
"""
if getattr(func, "__name__", "not lambda") == "<lambda>": # LambdaType does not work :(
warnings.warn("Lambda as constraint is not advice because it may not be picklable")
self._constraint_checkers.append(func)
Expand Down Expand Up @@ -327,6 +368,6 @@ def _set_random_state(self, random_state: np.random.RandomState) -> None:
if isinstance(param, Parameter):
param._set_random_state(random_state)

def complies_with_constraint(self) -> bool:
compliant = super().complies_with_constraint()
return compliant and all(param.complies_with_constraint() for param in self._parameters.values() if isinstance(param, Parameter))
def satisfies_constraint(self) -> bool:
compliant = super().satisfies_constraint()
return compliant and all(param.satisfies_constraint() for param in self._parameters.values() if isinstance(param, Parameter))
52 changes: 50 additions & 2 deletions nevergrad/parametrization/data.py
Original file line number Diff line number Diff line change
Expand Up @@ -54,6 +54,7 @@ def __init__(
sigma = Log(init=1.0, exponent=1.2, mutable_sigma=False) if mutable_sigma else 1.0
super().__init__(sigma=sigma, recombination="average")
self._value: np.ndarray = init if init is not None else np.zeros(shape)
self.integer = False
self.exponent: t.Optional[float] = None
self.bounds: t.Tuple[t.Optional[np.ndarray], t.Optional[np.ndarray]] = (None, None)
self.bound_transform: t.Optional[trans.BoundTransform] = None
Expand All @@ -65,6 +66,8 @@ def sigma(self) -> t.Union[np.ndarray, float]:

@property
def value(self) -> np.ndarray:
if self.integer:
return np.round(self._value) # type: ignore
return self._value

@value.setter
Expand All @@ -90,6 +93,25 @@ def sample(self: A) -> A:

def set_bounds(self: A, a_min: BoundValue = None, a_max: BoundValue = None,
method: str = "clipping", full_range_sampling: bool = False) -> A:
"""Bounds all real values into [a_min, a_max] using a provided method

Parameters
----------
a_min: float or None
minimum value
a_max: float or None
maximum value
method: str
"clipping", "constraint", "tanh" or "arctan"
full_range_sampling: bool
whether calling the "sample" method of the parameter should sample uniformly (or log-uniformly) on the whole
range of the bounds instead of sampling using a mutation on the current value

Notes
-----
- "tanh" reaches the boundaries really quickly, while "arctan" is much softer
- only "clipping" accepts partial bounds (None values)
""" # TODO improve description of methods
bounds = tuple(a if isinstance(a, np.ndarray) or a is None else np.array([a], dtype=float) for a in (a_min, a_max))
both_bounds = all(b is not None for b in bounds)
# preliminary checks
Expand Down Expand Up @@ -125,6 +147,22 @@ def set_bounds(self: A, a_min: BoundValue = None, a_max: BoundValue = None,
return self

def set_mutation(self: A, sigma: t.Optional[t.Union[float, "Array"]] = None, exponent: t.Optional[float] = None) -> A:
"""Output will be cast to integer(s) through deterministic rounding.

Parameters
----------
sigma: Array/Log or float
The standard deviation of the mutation. If a Parameter is provided, it will replace the current
value. If a float is provided, it will either replace a previous float value, or update the value
of the Parameter.
exponent: float
exponent for the logarithmic mode. With the default sigma=1, using exponent=2 will perform
x2 or /2 "on average" on the value at each mutation.

Returns
-------
self
"""
if sigma is not None:
# just replace if an actual Parameter is provided as sigma, else update value (parametrized or not)
if isinstance(sigma, Parameter) or isinstance(self.subparameters._parameters["sigma"], float):
Expand All @@ -142,6 +180,16 @@ def set_mutation(self: A, sigma: t.Optional[t.Union[float, "Array"]] = None, exp
self.exponent = exponent
return self

def set_integer_casting(self: A) -> A:
"""Output will be cast to integer(s) through deterministic rounding.

Returns
-------
self
"""
self.integer = True
return self

# pylint: disable=unused-argument
def _internal_set_std_data(self: A, data: np.ndarray, instance: A, deterministic: bool = True) -> A:
assert isinstance(data, np.ndarray)
Expand All @@ -156,7 +204,7 @@ def _internal_spawn_child(self) -> "Array":
child = self.__class__(init=self.value)
child.subparameters._parameters = {k: v.spawn_child() if isinstance(v, Parameter) else v
for k, v in self.subparameters._parameters.items()}
for name in ["exponent", "bounds", "bound_transform", "full_range_sampling"]:
for name in ["integer", "exponent", "bounds", "bound_transform", "full_range_sampling"]:
setattr(child, name, getattr(self, name))
return child

Expand Down Expand Up @@ -189,7 +237,7 @@ def __init__(self, init: float = 0.0, mutable_sigma: bool = True) -> None:

@property # type: ignore
def value(self) -> float: # type: ignore
return self._value[0] # type: ignore
return self._value[0] if not self.integer else int(self._value[0]) # type: ignore

@value.setter
def value(self, value: float) -> None:
Expand Down
12 changes: 7 additions & 5 deletions nevergrad/parametrization/test_parameter.py
Original file line number Diff line number Diff line change
Expand Up @@ -81,9 +81,9 @@ def test_parameters_basic_features(param: Parameter) -> None:
with pytest.warns(UserWarning):
param.register_cheap_constraint(lambda *args, **kwargs: False)
child2 = param.spawn_child()
assert child.complies_with_constraint()
assert not param.complies_with_constraint()
assert not child2.complies_with_constraint()
assert child.satisfies_constraint()
assert not param.satisfies_constraint()
assert not child2.satisfies_constraint()
# array to and from with hash
data_hash = param.get_data_hash()
param.set_std_data(param.get_std_data())
Expand All @@ -93,7 +93,7 @@ def test_parameters_basic_features(param: Parameter) -> None:
pickle.loads(string)
# array info transfer:
if isinstance(param, par.Array):
for name in ("exponent", "bounds", "bound_transform", "full_range_sampling"):
for name in ("integer", "exponent", "bounds", "bound_transform", "full_range_sampling"):
assert getattr(param, name) == getattr(child, name)


Expand Down Expand Up @@ -130,6 +130,8 @@ def test_scalar_and_mutable_sigma() -> None:
assert param.sigma == 5
param.mutate()
assert param.sigma != 5
param.set_integer_casting()
assert isinstance(param.value, int)


def test_array_recombination() -> None:
Expand All @@ -151,7 +153,7 @@ def test_constraints(name: str) -> None:
param.set_std_data(param.get_std_data())
np.testing.assert_approx_equal(param.value, 12, err_msg="Back and forth did not work")
param.set_std_data(np.array([100000.0]))
if param.complies_with_constraint():
if param.satisfies_constraint():
np.testing.assert_approx_equal(param.value, 100, significant=3, err_msg="Constraining did not work")


Expand Down