#11437 closed enhancement (fixed)
Configuration sections from plugins should be written to trac.ini
Reported by: | Ryan J Ollos | Owned by: | Ryan J Ollos |
---|---|---|---|
Priority: | normal | Milestone: | 1.1.4 |
Component: | general | Version: | |
Severity: | normal | Keywords: | configuration trac.ini |
Cc: | hasienda, Jun Omae | Branch: | |
Release Notes: |
|
||
API Changes: |
The |
||
Internal Changes: |
Description (last modified by )
When a Trac environment is first created the default data is written to trac.ini. However, when installing a plugin the default data is not written to trac.ini.
When the plugin enabled it would be nice to have the default data added to trac.ini. Currently the user must manually type the section and option name if they wish to change any configuration values. I think that most users would expect that the defaults for all configuration values be found in trac.ini.
A quick implementation that is not ideal, but demonstrates the idea is found in log:rjollos.git:t11437.
One issue with the quick implementation is that all of defaults are written to trac.ini, not just those of the plugin being installed. Maybe this is okay, but if a user likes to remove a lot default options to simplify the file, they may be surprised to have their changes undone (#7378).
#8290 had me thinking of a different solution, that plugins could add the information through existing IEnvironmentParticipant
method, or new methods or helper functions that we add. The downside of course is that every plugin then needs to be modified to implement the behavior.
Side note: it might be useful to have a trac-admin $ENV config clean
command to remove options from trac.ini that are no longer defined in an enabled Component. For example, with th:AccountManagerPlugin, many options were renamed going from 0.3 → 0.5, and the users must manually clean up these sections. This comes up rather frequently when users post their configuration to tickets and it contains a mess of old and new configuration options. Of course, it causes no harm to leave them, it is just more confusing and error prone to manage the trac.ini file. We could, by default, create a backup of the trac.ini file whenever clean is run.
Attachments (0)
Change History (25)
comment:1 by , 11 years ago
Cc: | added |
---|
comment:2 by , 11 years ago
Description: | modified (diff) |
---|---|
Milestone: | undecided → 1.0.2 |
follow-up: 8 comment:3 by , 11 years ago
FWIW, Please keep in mind the use of the "global configuration". I use this extensively, and I keep most settings in the global configuration file.
For instance, if updating some setting in a Trac instance introduced a lot of other kinds of settings into the local configuration file, it could be then necessary to clean out all the defaults overriding the global configuration.
(I mainly mention this due to the comment about writing all defaults to trac.ini, not just those of the plugin being installed.)
comment:4 by , 11 years ago
The latest changes after #11520 can be found in log:rjollos.git:t11437.2. However, I plan to do more work before committing.
follow-up: 6 comment:5 by , 11 years ago
I am not convinced of the usefulness of this. What happens if the configuration format for a plugin changes? Why would an administrator expect default options to be explicitly in the Trac configuration? The current system, where the administrator may set/override (default) configuration options works fine and is more robust in my opinion.
comment:6 by , 11 years ago
Replying to S.N.Maijers@…:
What happens if the configuration format for a plugin changes?
The same thing that would happen if the option has been added by editing the config file. The extra entries would be harmless and are the use case for the config clean
command that I describe in comment:description.
Why would an administrator expect default options to be explicitly in the Trac configuration?
Less typing, less possibility of mistakes in entering the option name and less need to look up what options are available in the documentation. Every default option for Trac is written to the config file, so adding plugin defaults would be consistent.
The current system, where the administrator may set/override (default) configuration options works fine and is more robust in my opinion.
You've not made an argument for why the change would not be robust.
comment:7 by , 11 years ago
Status: | new → assigned |
---|
comment:8 by , 11 years ago
Replying to anonymous:
FWIW, Please keep in mind the use of the "global configuration". I use this extensively, and I keep most settings in the global configuration file.
+1 Please take into account multi-project trac installations where
- there's one "master" project and all the "slave" projects inherit its global configuration, overriding only those config options that differ
- plugins are installed globally i.e. the Python installation or virtualenv used for running Trac, not in a project's
plugins
dir
Changes introduced with this ticket should not lead to writing defaults to "slave" trac.ini
files
comment:9 by , 11 years ago
Milestone: | 1.0.2 → 1.0.3 |
---|
comment:10 by , 10 years ago
Milestone: | 1.0.3 → 1.1.3 |
---|
comment:11 by , 10 years ago
Milestone: | 1.1.3 → 1.1.4 |
---|
comment:12 by , 10 years ago
Proposed changes in log:rjollos.git:t11437.4. The following behaviors have been implemented:
- When a plugin is enabled through TracAdmin or the Web administration page, the component options are written to trac.ini if they don't already exists in trac.ini or a parent configuration.
- The
# option-name = <inherited>
markers are no longer written to trac.ini, but still if the value is defined in a parent configuration, the default value is not written to trac.ini when enabling a component or creating an environment.
comment:13 by , 10 years ago
Release Notes: | modified (diff) |
---|
comment:14 by , 10 years ago
Reviewing.
- If possible, I think we should add unit tests for
Configuration.set_defaults(component=...)
with and without[inherit] file = ...
. - If the
component
argument istracopt.ticket.commit_updater.Commit
, the method would wrongly set defaults of options inCommitTicketUpdater
. My PoC is wrong.-
trac/config.py
diff --git a/trac/config.py b/trac/config.py index 6bcf5d7..f56a1dc 100644
a b class Configuration(object): 314 314 315 315 if component: 316 316 if component.endswith('.*'): 317 component = component[:- 1]318 component = component.lower() 317 component = component[:-2] 318 component = component.lower().split('.') 319 319 from trac.core import ComponentMeta 320 320 for cls in ComponentMeta._components: 321 clsname = (cls.__module__ + '.' + cls.__name__).lower() 322 if not clsname.startswith(component) and clsname != component: 321 clsname = (cls.__module__ + '.' + cls.__name__).lower() \ 322 .split('.') 323 if clsname[:len(component)] != component and \ 324 clsname != component: 323 325 continue 324 326 for option in cls.__dict__.itervalues(): 325 327 if not isinstance(option, Option):
-
comment:15 by , 10 years ago
Additional refactoring and tests added in log:rjollos.git:t11437.4. There are some test failures when running $ make test=trac/tests/__init__.py
, which are due to the other Components defined in trac.tests.config
. I'm thinking about how to modify or restructure the tests to avoid the failures, but I welcome ideas on that.
follow-up: 17 comment:16 by , 10 years ago
The Configuration.set_defaults
method uses registered classes in ComponentMeta._components
. I think we should clean up the class instances to remove test classes and interfaces in the tearDown
of unit tests. ConfigSection.registry
and ComponentMeta._registry
have the same issue. See [0ba577eb4/jomae.git].
Minor thing, component.rstrip('.*')
is used, Environment
class strips only the trailing .*
of component name in tags/trac-1.1.3/trac/env.py@:385#L382. I think we should use the same logic.
follow-ups: 18 19 comment:17 by , 10 years ago
Replying to jomae:
The
Configuration.set_defaults
method uses registered classes inComponentMeta._components
. I think we should clean up the class instances to remove test classes and interfaces in thetearDown
of unit tests.ConfigSection.registry
andComponentMeta._registry
have the same issue. See [0ba577eb4/jomae.git].
Thanks, I was trying to accomplish that but having no success.
If I run the tests using either of these two commands, ConfigurationSetDefaultsTestCase
is skipped:
$ PYTHONPATH=. make test=trac/tests/config.py $ PYTHONPATH=. python test/tests/config.py
SKIP: trac.tests.config.ConfigurationSetDefaultsTestCase (__name__ is not trac.tests.config)
Those two are the same of course because Makefile
runs python test/tests/config.py
. The tests get run for this command though:
$ python setup.py test -s trac.tests.config
Is that an expected limitation? If so, maybe we should modify how our Makefile
runs the tests.
Minor thing,
component.rstrip('.*')
is used,Environment
class strips only the trailing.*
of component name in tags/trac-1.1.3/trac/env.py@:385#L382. I think we should use the same logic.
Does that mean that trac.tests.config.
should not cause the components in the trac.tests.config
to be enabled? I suspect yes, since Trac enables the components in module1
with the following rules:
module1 = enabled
module1.* = enabled
but not with:
module1. = enabled
I guess that was probably your point though in saying that we should replicate the rule that exists in trac.env
.
comment:18 by , 10 years ago
Is that an expected limitation? If so, maybe we should modify how our
Makefile
runs the tests.
Yes. Running python trac/tests/config.py
, CompA
and CompB
classes are defined under __main__
package. Then, all tests of ConfigurationSetDefaultsTestCase
would fail. Similar discussion in comment:11:ticket:11312.
Minor thing,
component.rstrip('.*')
is used,Environment
class strips only the trailing.*
of component name in tags/trac-1.1.3/trac/env.py@:385#L382. I think we should use the same logic.Does that mean that
trac.tests.config.
should not cause the components in thetrac.tests.config
to be enabled? I suspect yes, since Trac enables the components inmodule1
with the following rules:
Right. modname1. = enabled
wouldn't enable components in modname1
package.
follow-up: 20 comment:19 by , 10 years ago
Those two are the same of course because
Makefile
runspython test/tests/config.py
. The tests get run for this command though:$ python setup.py test -s trac.tests.config
After the following patch, the Makefile would convert the file name to module name. We could run unit tests with make test=trac/tests/config.py
and make trac/tests/config.py
.
-
Makefile
diff --git a/Makefile b/Makefile index dcee889..7eef4d7 100644
a b export HELP_CFG 124 124 125 125 .PHONY: all help status clean clean-bytecode clean-mo 126 126 127 %.py : status 128 python -m $(subst /,.,$(@:.py=)) $(testopts) 129 127 130 ifdef test 128 131 all: status 129 python $(test) $(testopts)132 python -m $(subst /,.,$(test:.py=)) $(testopts) 130 133 else 131 134 all: help 132 135 endif
follow-up: 21 comment:20 by , 10 years ago
Replying to jomae:
After the following patch, the Makefile would convert the file name to module name. We could run unit tests with
make test=trac/tests/config.py
andmake trac/tests/config.py
.
The ConfigurationSetDefaultsTestCase
tests are still skipped since __name__ == __main__
in the module. Is there another complementary change that I should make?
With the following variation of your patch the ConfigurationSetDefaultsTestCase
are executed using make test=trac/tests/config.py
and make trac/tests/config.py
.
-
Makefile
diff --git a/Makefile b/Makefile index 4c5d3a0..83f82c6 100644
a b export HELP_CFG 125 125 .PHONY: all help status clean clean-bytecode clean-mo 126 126 127 127 %.py : status 128 python setup.py -q test -s $(subst /,.,$(@:.py=)) $(testopts)128 python setup.py -q test -s $(subst /,.,$(@:.py=)).suite $(testopts) 129 129 130 130 ifdef test 131 131 all: status 132 python setup.py -q test -s $(subst /,.,$(test:.py=)) $(testopts)132 python setup.py -q test -s $(subst /,.,$(test:.py=)).suite $(testopts) 133 133 else 134 134 all: help 135 135 endif
I haven't done enough testing yet to be confident of the change.
comment:21 by , 10 years ago
Replying to rjollos:
The
ConfigurationSetDefaultsTestCase
tests are still skipped since__name__ == __main__
in the module. Is there another complementary change that I should make?
Sorry. I was confused with something else.
With the following variation of your patch the
ConfigurationSetDefaultsTestCase
are executed usingmake test=trac/tests/config.py
andmake trac/tests/config.py
.[…]
I haven't done enough testing yet to be confident of the change.
That patch seems to be good to me.
comment:22 by , 10 years ago
Thanks for the review. I committed against #11316 since the change fixes the issue in that ticket as well. I'm thinking we should suggest in TracDev/UnitTests to use the Makefile for test execution rather than suggest so many different ways to execute the tests, some of which can fail for some modules.
I'm also wondering if there might be a better way yet. If we use python -m unittest
to run the tests, we could directly specify a single class or method to execute.
$ PYTHONPATH=. python -m unittest trac.tests.config ............................................. ---------------------------------------------------------------------- Ran 45 tests in 1.621s OK $ PYTHONPATH=. python -m unittest trac.tests.config.ConfigurationTestCase ...................................... ---------------------------------------------------------------------- Ran 38 tests in 2.068s OK $ PYTHONPATH=. python -m unittest trac.tests.config.ConfigurationTestCase.test_repr . ---------------------------------------------------------------------- Ran 1 test in 0.017s OK
-
Makefile
diff --git a/Makefile b/Makefile index 83f82c6..600325a 100644
a b export HELP_CFG 125 125 .PHONY: all help status clean clean-bytecode clean-mo 126 126 127 127 %.py : status 128 python setup.py -q test -s $(subst /,.,$(@:.py=)).suite$(testopts)128 python -m unittest $(subst /,.,$(@:.py=)) $(testopts) 129 129 130 130 ifdef test 131 131 all: status 132 python setup.py -q test -s $(subst /,.,$(test:.py=)).suite$(testopts)132 python -m unittest $(subst /,.,$(test:.py=)) $(testopts) 133 133 else 134 134 all: help 135 135 endif
However, there are test failures when running trac/wiki/tests/formatter.py
, and no tests run when specifying trac/wiki/tests/__init__.py
. Perhaps that is worth investigating more down the road.
comment:24 by , 10 years ago
API Changes: | modified (diff) |
---|---|
Release Notes: | modified (diff) |
Resolution: | → fixed |
Status: | assigned → closed |
Committed to trunk in [13859:13860].
comment:25 by , 10 years ago
Replying to rjollos:
Side note: it might be useful to have a
trac-admin $ENV config clean
command to remove options from trac.ini that are no longer defined in an enabled Component.
Related idea in comment:15:ticket:6551.
Another PoC: jomae.git@t11437.1.