summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorjan.nijtmans <nijtmans@users.sourceforge.net>2024-03-10 13:04:36 (GMT)
committerjan.nijtmans <nijtmans@users.sourceforge.net>2024-03-10 13:04:36 (GMT)
commit24245629354b88bd86167e9f8b8f8a9170dfe5bd (patch)
treefeeb4afa0b424d88feaa912a39297efc68bbfdd9
parent57ec55775381504d05818dab74817965c9fe9961 (diff)
parentb39d8850d6ddcc5e80eb47d1af7f34d19cbbcb22 (diff)
downloadtcl-24245629354b88bd86167e9f8b8f8a9170dfe5bd.zip
tcl-24245629354b88bd86167e9f8b8f8a9170dfe5bd.tar.gz
tcl-24245629354b88bd86167e9f8b8f8a9170dfe5bd.tar.bz2
Fix [910d67a229fe7f65]: NS-qualified invocations of command evade [namespace unknown]
-rw-r--r--generic/tclBasic.c51
-rw-r--r--tests/namespace.test42
2 files changed, 89 insertions, 4 deletions
diff --git a/generic/tclBasic.c b/generic/tclBasic.c
index e8ff3d9..592529f 100644
--- a/generic/tclBasic.c
+++ b/generic/tclBasic.c
@@ -4805,11 +4805,54 @@ TEOV_NotFound(
* namespace (TIP 181). */
Namespace *savedNsPtr = NULL;
+ Tcl_Size qualLen;
+ const char *qualName = Tcl_GetStringFromObj(objv[0], &qualLen);
+
currNsPtr = varFramePtr->nsPtr;
- if ((currNsPtr == NULL) || (currNsPtr->unknownHandlerPtr == NULL)) {
- currNsPtr = iPtr->globalNsPtr;
- if (currNsPtr == NULL) {
- Tcl_Panic("Tcl_EvalObjv: NULL global namespace pointer");
+ if ((currNsPtr == NULL) || (currNsPtr->unknownHandlerPtr == NULL) ||
+ (qualLen > 2 && memchr(qualName, ':', qualLen)) /* fast check for NS:: */
+ ) {
+ /*
+ * first try to find namespace unknown handler of the namespace
+ * of executed command if available:
+ */
+ Namespace *altNsPtr, *dummyNsPtr;
+ const char *simpleName;
+
+ (void) TclGetNamespaceForQualName(interp, qualName, currNsPtr,
+ TCL_FIND_IF_NOT_SIMPLE, &currNsPtr, &altNsPtr,
+ &dummyNsPtr, &simpleName);
+ if (!simpleName) {
+ goto globNS;
+ }
+ if (!currNsPtr || (currNsPtr == iPtr->globalNsPtr)) {
+ if (!altNsPtr || (altNsPtr == iPtr->globalNsPtr)) {
+ goto globNS;
+ }
+ currNsPtr = altNsPtr;
+ }
+ while (currNsPtr->unknownHandlerPtr == NULL ||
+ (currNsPtr->flags & (NS_DYING | NS_DEAD))
+ ) {
+ /* traverse to alive parent namespace containing handler */
+ if (!(currNsPtr = currNsPtr->parentPtr) ||
+ (currNsPtr == iPtr->globalNsPtr)
+ ) {
+ /* continue from alternate NS if available */
+ if (!altNsPtr || (altNsPtr == iPtr->globalNsPtr)) {
+ goto globNS;
+ }
+ currNsPtr = altNsPtr;
+ altNsPtr = NULL;
+ continue;
+ globNS:
+ /* fallback to the global unknown */
+ currNsPtr = iPtr->globalNsPtr;
+ if (currNsPtr == NULL) {
+ Tcl_Panic("TEOV_NotFound: NULL global namespace pointer");
+ }
+ break;
+ }
}
}
diff --git a/tests/namespace.test b/tests/namespace.test
index ae233cb..abe642e 100644
--- a/tests/namespace.test
+++ b/tests/namespace.test
@@ -3124,6 +3124,48 @@ test namespace-52.12 {unknown: error case must not reset handler} -body {
} -cleanup {
namespace delete foo
} -result ok
+test namespace-52.13 {unknown: invocation outside of NS doesn't evade namespace unknown, bug 910d67a229fe7f65} -body {
+ namespace eval ::foo::bar {
+ proc _unknown args {list ::foo:bar:_unknown [uplevel {namespace current}] $args}
+ namespace unknown [namespace current]::_unknown
+ }
+ list [namespace inscope ::foo::bar {xxx}] [namespace inscope ::foo {bar::xxx}] [::foo::bar::xxx] [namespace inscope :: {foo::bar::xxx}]
+} -cleanup {
+ namespace delete ::foo
+} -result {{::foo:bar:_unknown ::foo::bar xxx} {::foo:bar:_unknown ::foo bar::xxx} {::foo:bar:_unknown :: ::foo::bar::xxx} {::foo:bar:_unknown :: foo::bar::xxx}}
+test namespace-52.14 {unknown: invocation outside of NS doesn't evade namespace unknown for command with sub-NS, bug 910d67a229fe7f65} -body {
+ namespace eval ::foo::bar {
+ proc _unknown args {list ::foo:bar:_unknown [uplevel {namespace current}] $args}
+ namespace unknown [namespace current]::_unknown
+ }
+ set res {}
+ lappend res [namespace inscope ::foo::bar {xxx::yyy}] [namespace inscope ::foo {bar::xxx::yyy}] [::foo::bar::xxx::yyy] [namespace inscope :: {foo::bar::xxx::yyy}]
+ # now with existsing ::foo::bar::xxx, but without unknown handler inside (only parent ::foo::bar has a handler):
+ namespace eval ::foo::bar::xxx {}
+ lappend res [namespace inscope ::foo::bar {xxx::yyy}] [namespace inscope ::foo {bar::xxx::yyy}] [::foo::bar::xxx::yyy] [namespace inscope :: {foo::bar::xxx::yyy}]
+} -cleanup {
+ namespace delete ::foo
+ unset -nocomplain res
+} -result [lrepeat 2 \
+ {::foo:bar:_unknown ::foo::bar xxx::yyy} {::foo:bar:_unknown ::foo bar::xxx::yyy} {::foo:bar:_unknown :: ::foo::bar::xxx::yyy} {::foo:bar:_unknown :: foo::bar::xxx::yyy}
+]
+test namespace-52.14 {unknown: it must consider alternate search path (relative global NS), bug 910d67a229fe7f65} -body {
+ namespace eval ::foo::bar {}
+ namespace eval ::xxx::yyy {
+ proc _unknown args {list ::xxx:yyy:_unknown [uplevel {namespace current}] $args}
+ namespace unknown [namespace current]::_unknown
+ }
+ set res {}
+ lappend res [namespace inscope ::foo::bar {xxx::yyy::cmd}] [namespace inscope ::foo {xxx::yyy::cmd}]
+ namespace eval ::foo::bar::xxx {}
+ lappend res [namespace inscope ::foo::bar {xxx::yyy::cmd}] [namespace inscope ::foo {xxx::yyy::cmd}]
+ namespace eval ::foo::bar::xxx::yyy {}
+ lappend res [namespace inscope ::foo::bar {xxx::yyy::cmd}] [namespace inscope ::foo {xxx::yyy::cmd}]
+} -cleanup {
+ namespace delete ::foo
+ namespace delete ::xxx
+ unset -nocomplain res
+} -result [lrepeat 3 {::xxx:yyy:_unknown ::foo::bar xxx::yyy::cmd} {::xxx:yyy:_unknown ::foo xxx::yyy::cmd}]
# TIP 314 - ensembles with parameters
test namespace-53.1 {ensembles: parameters} {