Browse Source

whitespace (end of line) fixes and fconfigure->chan configure fileevent->chan event, minor expr tweaks

master
Julian Noble 3 weeks ago
parent
commit
fe03c0652b
  1. 22
      src/modules/#modpod-zipper-999999.0a1.0/zipper-999999.0a1.0.tm
  2. 4
      src/modules/patternpunk-1.1.tm
  3. 735
      src/modules/punk-0.1.tm
  4. 28
      src/modules/punk/aliascore-999999.0a1.0.tm
  5. 764
      src/modules/punk/ansi-999999.0a1.0.tm
  6. 584
      src/modules/punk/args-999999.0a1.0.tm
  7. 140
      src/modules/punk/args/tclcore-999999.0a1.0.tm
  8. 42
      src/modules/punk/assertion-999999.0a1.0.tm
  9. 200
      src/modules/punk/basictelnet-999999.0a1.0.tm
  10. 60
      src/modules/punk/cap-999999.0a1.0.tm
  11. 4
      src/modules/punk/cap/handlers/caphandler-999999.0a1.0.tm
  12. 78
      src/modules/punk/cap/handlers/templates-999999.0a1.0.tm
  13. 2
      src/modules/punk/char-999999.0a1.0.tm
  14. 2
      src/modules/punk/config-0.1.tm
  15. 154
      src/modules/punk/console-999999.0a1.0.tm
  16. 203
      src/modules/punk/fileline-999999.0a1.0.tm
  17. 30
      src/modules/punk/icomm-999999.0a1.0.tm
  18. 128
      src/modules/punk/imap4-999999.0a1.0.tm
  19. 574
      src/modules/punk/lib-999999.0a1.0.tm
  20. 100
      src/modules/punk/mix/base-0.1.tm
  21. 118
      src/modules/punk/mix/cli-999999.0a1.0.tm
  22. 6
      src/modules/punk/mix/templates-999999.0a1.0.tm
  23. 12
      src/modules/punk/mix/util-999999.0a1.0.tm
  24. 176
      src/modules/punk/nav/fs-999999.0a1.0.tm
  25. 24
      src/modules/punk/repl-999999.0a1.0.tm
  26. 38
      src/modules/punk/repl/codethread-999999.0a1.0.tm
  27. 56
      src/modules/punk/sshrun-999999.0a1.0.tm
  28. 84
      src/modules/punk/winrun-999999.0a1.0.tm
  29. 128
      src/modules/punkcheck-0.1.0.tm
  30. 85
      src/modules/shellrun-0.1.1.tm
  31. 106
      src/modules/shellthread-1.6.1.tm
  32. 6
      src/modules/tcl9test-999999.0a1.0.tm
  33. 936
      src/modules/textblock-999999.0a1.0.tm

22
src/modules/#modpod-zipper-999999.0a1.0/zipper-999999.0a1.0.tm

@ -10,7 +10,7 @@ namespace eval zipper {
variable base
variable toc
}
#if we initialize before writing anything to fd - our base is the file base
# - ie we get an 'internal preamble'
#if instead, we write data to fd before initialize, our base is the start of the archive-data.
@ -22,7 +22,7 @@ namespace eval zipper {
#
#It seems to be ok either way for reading - but some tools cannot write to file based offset if there is prefix data
#(e.g file.kit with offset adjusted with something like zip -A which makes the preamble internal to the zip)
# and some cannot write to archive-based offset if there is prefix data !
# and some cannot write to archive-based offset if there is prefix data !
#(e.g file.kit with preamble prepended and offsets not adjusted = external preamble)
#
#Some tools may auto-adjust to file-based offset when adding entries (e.g pkzip if extension is .zip)
@ -31,8 +31,8 @@ namespace eval zipper {
set v::fd $fd
set v::base [tell $fd]
set v::toc {}
#fconfigure $fd -translation binary -encoding binary
fconfigure $fd -translation binary -encoding iso8859-1
#chan configure $fd -translation binary -encoding binary
chan configure $fd -translation binary -encoding iso8859-1
}
proc emit {s} {
@ -83,7 +83,7 @@ namespace eval zipper {
#lappend v::toc "[binary format a2c6ssssiiiss4ii PK {1 2 20 0 20 0} \
# $flag $type $time $date $crc $csize $fsize $fnlen \
# {0 0 0 0} 128 [tell $v::fd]]$name"
#build the CDR file header - but we don't add it here
#build the CDR file header - but we don't add it here
set do_extended_timestamp 1
if {!$do_extended_timestamp} {
lappend v::toc "[binary format a2c6ssssiiiss4ii PK {1 2 20 0 20 0} \
@ -100,7 +100,7 @@ namespace eval zipper {
# (ModTime) Long time of last modification (UTC/GMT)
# ---
# - Tsize = 9 - 4 = 5
set extended_timestamp [binary format a2sci UT 5 0 $unixmtime]
set extended_timestamp [binary format a2sci UT 5 0 $unixmtime]
append extra $extended_timestamp
# ---
@ -127,13 +127,13 @@ namespace eval zipper {
set fsize 0
set csize 0
set fnlen [string length $name]
set crc 0
lappend v::toc "[binary format a2c6ssssiiiss4ii PK {1 2 20 0 20 0} \
$flag $type $time $date $crc $csize $fsize $fnlen \
{0 0 0 0} 128 [tell $v::fd]]$name"
emit [binary format a2c4ssssiiiss PK {3 4 20 0} \
$flag $type $time $date $crc $csize $fsize $fnlen 0]
emit $name
@ -150,7 +150,7 @@ namespace eval zipper {
set len [expr {$cd_end_pos - $cd_start_pos}]
#incr pos -$v::base
set cdr_offset_pos [expr $cd_start_pos -$v::base] ;#review
set cdr_offset_pos [expr {$cd_start_pos -$v::base}] ;#review
#EOCD signature PK\5\6 = 0x06054b50
emit [binary format a2c2ssssiis PK {5 6} 0 0 $ntoc $ntoc $len $cdr_offset_pos 0]
@ -178,7 +178,7 @@ if {[info exists argv0] && [string match zipper-* [file tail $argv0]]} {
if {[file isfile $f]} {
regsub {^\./} $f {} f
set fd [open $f]
fconfigure $fd -translation binary -encoding binary
chan configure $fd -translation binary -encoding binary
zipper::addentry $f [read $fd] [file mtime $f]
close $fd
} elseif {[file isdir $f]} {

4
src/modules/patternpunk-1.1.tm

@ -117,7 +117,7 @@ punk::args::define {
@cmd -name ">punk . poses" -help "Show or list the poses for the Punk mascot"
-censored -default 1 -type boolean -help "Set true to include mild toilet humour poses"
-return -default table -choices {list table}
}
}
>punk .. Method poses {args} {
set argd [punk::args::get_by_id ">punk . poses" $args]
set censored [dict get $argd opts -censored]
@ -424,7 +424,7 @@ namespace eval patternpunk::lib {
proc K {x y} {return $x}
}
package provide patternpunk [namespace eval patternpunk {
variable version
variable version
set version 1.1
}]

735
src/modules/punk-0.1.tm

File diff suppressed because it is too large Load Diff

28
src/modules/punk/aliascore-999999.0a1.0.tm

@ -21,7 +21,7 @@
#[manpage_begin punkshell_module_punk::aliascore 0 999999.0a1.0]
#[copyright "2024"]
#[titledesc {punkshell command aliases}] [comment {-- Name section and table of contents description --}]
#[moddesc {-}] [comment {-- Description at end of page heading --}]
#[moddesc {-}] [comment {-- Description at end of page heading --}]
#[require punk::aliascore]
#[keywords module alias]
#[description]
@ -98,7 +98,7 @@ package require Tcl 8.6-
# Base namespace
# ++ +++ +++ +++ +++ +++ +++ +++ +++ +++ +++
tcl::namespace::eval punk::aliascore {
tcl::namespace::export {[a-z]*} ;# Convention: export all lowercase
tcl::namespace::export {[a-z]*} ;# Convention: export all lowercase
variable aliases
#use absolute ns ie must be prefixed with ::
#single element commands are imported if source command already exists, otherwise aliased. multi element commands are aliased
@ -136,7 +136,7 @@ tcl::namespace::eval punk::aliascore {
#*** !doctools
#[subsection {Namespace punk::aliascore}]
#[para] Core API functions for punk::aliascore
#[para] Core API functions for punk::aliascore
#[list_begin definitions]
@ -144,13 +144,13 @@ tcl::namespace::eval punk::aliascore {
#proc sample1 {p1 n args} {
# #*** !doctools
# #[call [fun sample1] [arg p1] [arg n] [opt {option value...}]]
# #[para]Description of sample1
# #[para]Description of sample1
# #[para] Arguments:
# # [list_begin arguments]
# # [arg_def tring p1] A description of string argument p1.
# # [arg_def integer n] A description of integer argument n.
# # [list_end]
# return "ok"
# return "ok"
#}
#todo - options as to whether we should raise an error if collisions found, undo aliases etc?
@ -208,13 +208,13 @@ tcl::namespace::eval punk::aliascore {
#todo - ensure exported? noclobber?
if {[tcl::namespace::tail $a] eq [tcl::namespace::tail $cmd]} {
#puts stderr "importing $cmd"
tcl::namespace::eval :: [list namespace import $cmd]
tcl::namespace::eval :: [list namespace import $cmd]
} else {
#target command name differs from exported name
#e.g stripansi -> punk::ansi::ansistrip
#import and rename
#puts stderr "importing $cmd (with rename to ::$a)"
tcl::namespace::eval $tempns [list namespace import $cmd]
tcl::namespace::eval $tempns [list namespace import $cmd]
catch {rename ${tempns}::[namespace tail $cmd] ::$a}
}
} else {
@ -242,18 +242,18 @@ tcl::namespace::eval punk::aliascore {
# Secondary API namespace
# ++ +++ +++ +++ +++ +++ +++ +++ +++ +++ +++
namespace eval punk::aliascore::lib {
namespace export {[a-z]*} ;# Convention: export all lowercase
namespace export {[a-z]*} ;# Convention: export all lowercase
namespace path [namespace parent]
#*** !doctools
#[subsection {Namespace punk::aliascore::lib}]
#[para] Secondary functions that are part of the API
#[para] Secondary functions that are part of the API
#[list_begin definitions]
#proc utility1 {p1 args} {
# #*** !doctools
# #[call lib::[fun utility1] [arg p1] [opt {?option value...?}]]
# #[para]Description of utility1
# return 1
# #[para]Description of utility1
# return 1
#}
@ -271,17 +271,17 @@ namespace eval punk::aliascore::lib {
namespace eval punk::aliascore::system {
#*** !doctools
#[subsection {Namespace punk::aliascore::system}]
#[para] Internal functions that are not part of the API
#[para] Internal functions that are not part of the API
}
# ++ +++ +++ +++ +++ +++ +++ +++ +++ +++ +++
## Ready
## Ready
package provide punk::aliascore [namespace eval punk::aliascore {
variable pkg punk::aliascore
variable version
set version 999999.0a1.0
set version 999999.0a1.0
}]
return

764
src/modules/punk/ansi-999999.0a1.0.tm

File diff suppressed because it is too large Load Diff

584
src/modules/punk/args-999999.0a1.0.tm

File diff suppressed because it is too large Load Diff

140
src/modules/punk/args/tclcore-999999.0a1.0.tm

@ -21,7 +21,7 @@
#[manpage_begin punkshell_module_punk::args::tclcore 0 999999.0a1.0]
#[copyright "2025"]
#[titledesc {punk::args definitions for tcl core commands}] [comment {-- Name section and table of contents description --}]
#[moddesc {tcl core argument definitions}] [comment {-- Description at end of page heading --}]
#[moddesc {tcl core argument definitions}] [comment {-- Description at end of page heading --}]
#[require punk::args::tclcore]
#[keywords module]
#[description]
@ -117,7 +117,7 @@ tcl::namespace::eval punk::args::tclcore {
# set A_RST "\x1b\[0m"
#}
#we can't just strip ansi as there are non colour codes such as hyperlink that should be maintained whether color is on or off.
#we can't just strip ansi as there are non colour codes such as hyperlink that should be maintained whether color is on or off.
#for now we can use reverse - (like underline, is a non-colour attribute that remains effective when color off in punk::ansi)
set A_WARN \x1b\[7m
set A_RST \x1b\[0m
@ -145,7 +145,7 @@ tcl::namespace::eval punk::args::tclcore {
tcl::namespace::import ::punk::ansi::a+
# -- --- --- --- ---
#non colour SGR codes
# we can use these directly via ${$I} etc without marking a definition with @dynamic
# we can use these directly via ${$I} etc without marking a definition with @dynamic
#This is because they don't need to change when colour switched on and off.
set I [a+ italic]
set NI [a+ noitalic]
@ -168,9 +168,9 @@ tcl::namespace::eval punk::args::tclcore {
by groupname. Each groupname forms the title of a subtable
in the choices list.
Subcommands not assigned to a groupname will appear first
in an untitled subtable."
in an untitled subtable."
-columns -default 4 -type integer -help\
"Max number of columns for all subtables in the choices
"Max number of columns for all subtables in the choices
display area"
@values -min 1 -max 1
ensemble -optional 0 -help\
@ -249,7 +249,7 @@ tcl::namespace::eval punk::args::tclcore {
puts --------------------
}
set opt_groupdict $checked_groupdict
set opt_groupdict $checked_groupdict
# ----------------------------------------------
set allgrouped [list]
dict for {g members} $opt_groupdict {
@ -271,7 +271,7 @@ tcl::namespace::eval punk::args::tclcore {
}
append argdef " \} -choicecolumns $opt_columns" \n
#todo -choicelabels
#todo -choicelabels
#detect subcommand further info available e.g if oo or ensemble or punk::args id exists..
#consider a different mechanism to add a label on rhs of same line as choice (for (i) marker)
@ -318,7 +318,7 @@ tcl::namespace::eval punk::args::tclcore {
"milliseconds"
@values -form {delay} -min 1 -max 1
@values -form {schedule_ms} -min 2
script -form {schedule_ms} -multiple 1 -optional 1 ref-help common_script_help
script -form {schedule_ms} -multiple 1 -optional 1 ref-help common_script_help
@form -form {cancelid} -synopsis "after cancel id"
@ -330,7 +330,7 @@ tcl::namespace::eval punk::args::tclcore {
@form -form {cancelscript} -synopsis "after cancel script ?script...?"
@values -min 2
cancel -choices {cancel}
script -multiple 1 -optional 0 ref-help common_script_help
script -multiple 1 -optional 0 ref-help common_script_help
@form -form {schedule_idle} -synopsis "after idle script ?script...?"
@ -365,13 +365,13 @@ tcl::namespace::eval punk::args::tclcore {
"Information about the state of the Tcl interpreter"
@leaders -min 1 -max 1
${[punk::args::tclcore::argdoc::info_subcommands]}
@values -min 0
@values -min 0
} "@doc -name Manpage: -url [manpage_tcl array]" ]
#An idiom for sharing common features - incomplete - todo work out what happens with (default)::id that has leaders,opts,values
#An idiom for sharing common features - incomplete - todo work out what happens with (default)::id that has leaders,opts,values
#todo @cmd -help+ text (append to existing help that came from a default?)
lappend PUNKARGS [list {
@id -id "(default)::tcl::binary::*::base64"
@ -416,14 +416,14 @@ tcl::namespace::eval punk::args::tclcore {
lappend PUNKARGS [list {
@id -id "::tcl::binary::encode::hex"
@default -id (default)::tcl::binary::*::hex
@cmd -name "binary encode hex"
@cmd -name "binary encode hex"
@values -min 1 -max 1
data -type string
} ]
lappend PUNKARGS [list {
@id -id "::tcl::binary::decode::hex"
@default -id (default)::tcl::binary::*::hex
@cmd -name "binary encode hex"
@cmd -name "binary encode hex"
-strict -type none -help\
"Instructs the decoder to throw an error if it encounters whitespace
characters. Otherwise it ignores them."
@ -445,10 +445,10 @@ tcl::namespace::eval punk::args::tclcore {
@id -id "::tcl::binary::encode::uuencode"
@default -id (default)::tcl::binary::*::uuencode
#todo @cmd -help+ "Changing the options may produce files that other implementations of decoders cannot process"
@cmd -name "binary encode uuencode"
@cmd -name "binary encode uuencode"
-maxlen -type integer -default 61 -range {5 85} -help\
"Indicates the maximum number of characters to produce for each encoded line.
The valid range is 5 to 85. Line lengths outside that range cannot be
The valid range is 5 to 85. Line lengths outside that range cannot be
accommodated by the encoding format."
-wrapchar -type string -default \n -help\
"Indicates the character(s) to use to mark the end of each encoded line.
@ -464,7 +464,7 @@ tcl::namespace::eval punk::args::tclcore {
lappend PUNKARGS [list {
@id -id "::tcl::binary::decode::uuencode"
@default -id (default)::tcl::binary::*::uuencode
@cmd -name "binary decode uuencode"
@cmd -name "binary decode uuencode"
-strict -type none -help\
"Instructs the decoder to throw an error if it encounters anything outside
of the standard encoding format. Without this option, the decoder tolerates
@ -540,14 +540,14 @@ tcl::namespace::eval punk::args::tclcore {
${$B}import${$N}
${$I}commandName${$NI} was created by 'namespace import'.
${$B}native${$N}
${$I}commandName${$NI} was created by the 'Tcl_CreateObjCommand' interface
${$I}commandName${$NI} was created by the 'Tcl_CreateObjCommand' interface
directly without further registration of the type of command.
${$B}object${$N}
${$I}commandName${$NI} is the public comand that represents an instance
of oo::object or one of its subclasses.
${$B}privateObject${$N}
${$I}commandName${$NI} is the private command, my by default,
that represents an instance of oo::object or one of its subclasses.
that represents an instance of oo::object or one of its subclasses.
${$B}proc${$N}
${$I}commandName${$NI} was created by 'proc'.
${$B}interp${$N}
@ -583,7 +583,7 @@ tcl::namespace::eval punk::args::tclcore {
If namespaceList is specified as a list of named namespaces, the current
namespace's command resolution path is set to those namespaces and returns
the empty list. The default command resolution path is always empty.
See the section NAME_RESOLUTION in the manpage for an explanation of the
See the section NAME_RESOLUTION in the manpage for an explanation of the
rules regarding name resolution."
@values -min 0 -max 1
namespaceList -type list -optional 1 -help\
@ -618,10 +618,10 @@ tcl::namespace::eval punk::args::tclcore {
regarding name resolution.
"
@opts
-command
-command
-variable
@values -min 1 -max 1
name
name
} "@doc -name Manpage: -url [manpage_tcl namespace]" ]
# -- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- ---
@ -631,9 +631,9 @@ tcl::namespace::eval punk::args::tclcore {
"Returns a dictionary mapping subprocess PIDs to their respective status.
If ${$I}pids${$NI} is specified as a list of PIDs then the command
only returns the status of the matching subprocesses if they exist.
For active processes, the status is an empty value. For terminated
For active processes, the status is an empty value. For terminated
processes, the status is a list with the following format:
{code ?msg errorCode?}
{code ?msg errorCode?}
where:
${$I}code${$NI}
is a standard Tcl return code, ie.,
@ -642,12 +642,12 @@ tcl::namespace::eval punk::args::tclcore {
is the human readable error message,
${$I}errorCode${$NI}
uses the same format as the errorCode global variable
Note that msg and errorCode are only present for abnormally
Note that msg and errorCode are only present for abnormally
terminated processes (i.e. those where the code is nonzero).
Under the hood this command calls Tcl_WaitPid with the
WNOHANG flag set for non-blocking behaviour, unless the -wait
switch is set (see below).
"
-wait -type none -optional 1 -help\
"By default the command returns immediately (the underlying Tcl_WaitPid
@ -680,7 +680,7 @@ tcl::namespace::eval punk::args::tclcore {
############################################################################################################################################################
# -- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- ---
# -- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- ---
# COMMANDS A-H
# COMMANDS A-H
# -- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- ---
# -- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- ---
############################################################################################################################################################
@ -699,7 +699,7 @@ tcl::namespace::eval punk::args::tclcore {
return [ensemble_subcommands_definition -groupdict $groups -columns 4 array]
}
}
lappend PUNKARGS [list {
@dynamic
@id -id ::array
@ -720,7 +720,7 @@ tcl::namespace::eval punk::args::tclcore {
This command is normally used within a procedure body (or method body,
or lambda term) to create a constant within that procedure, or within a
namespace eval body to create a constant within that namespace. The
namespace eval body to create a constant within that namespace. The
constant is an unmodifiable variable, called varName, that is initialised
with value. The result of const is always the empty string on success.
If a variable varname does not exist, it is create with its value set to
@ -733,7 +733,7 @@ tcl::namespace::eval punk::args::tclcore {
The varName may not be a qualified name or reference an element of an
array by any means. If the variable exists and is an array, that is an
error. Constants are normally only removed by their containing procedure
exiting or their namespace being deleted.
exiting or their namespace being deleted.
"
@values -min 1 -max 2
varName -help ""
@ -778,7 +778,7 @@ tcl::namespace::eval punk::args::tclcore {
@cmd -name "builtin: lappend" -help\
"Append list elements onto a variable.
"
@values -min 1 -max -1
@values -min 1 -max -1
varName -type string -help\
"variable name"
value -type any -optional 1 -multiple 1
@ -787,11 +787,11 @@ tcl::namespace::eval punk::args::tclcore {
# -- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- ---
punk::args::define {
@id -id ::ledit
@id -id ::ledit
@cmd -name "builtin: ledit" -help\
"Replace elements of a list stored in variable
"
@values -min 3 -max -1
@values -min 3 -max -1
listVar -type string -help\
"Existing list variable name"
first -type indexexpression
@ -804,7 +804,7 @@ tcl::namespace::eval punk::args::tclcore {
punk::args::define {
@id -id ::lremove
@cmd -name "builtin: lremove" -help\
"Remove elements from a list by index
"Remove elements from a list by index
lremove returns a new list formed by simultaneously removing zero or
more elements of list at each of the indices given by an arbitrary
number of index arguments. The indices may be in any order and may be
@ -813,7 +813,7 @@ tcl::namespace::eval punk::args::tclcore {
'string index', supporting simple index arithmetic and indices relative
to the end of the list. 0 refers to the first element of the list, and
end refers to the last element of the list."
@values -min 1 -max -1
@values -min 1 -max -1
list -type list -help\
"tcl list as a value"
index -type indexexpression -multiple 1 -optional 1
@ -824,11 +824,11 @@ tcl::namespace::eval punk::args::tclcore {
# -- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- ---
punk::args::define {
@id -id ::lpop
@id -id ::lpop
@cmd -name "builtin: lpop" -help\
"Get and remove an element in a list
"
@values -min 1 -max -1
@values -min 1 -max -1
varName -type string -help\
"Existing list variable name"
index -type indexexpression -default end -optional 1 -multiple 1 -help\
@ -866,7 +866,7 @@ tcl::namespace::eval punk::args::tclcore {
############################################################################################################################################################
# -- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- ---
# -- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- ---
# COMMANDS M-Z
# COMMANDS M-Z
# -- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- ---
# -- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- ---
############################################################################################################################################################
@ -882,19 +882,19 @@ tcl::namespace::eval punk::args::tclcore {
then set the value of varName to value, creating a new variable
if one does not already exist, and return its value. If varName
contains an open parenthesis and ends with a close parenthesis,
then it refers to an array element: the characters before the
first open parenthesis are the name of the array, and the
then it refers to an array element: the characters before the
first open parenthesis are the name of the array, and the
characters between the parentheses are the index within the array.
Otherwise varName refers to a scalar variable.
If varName includes namespace qualifiers (in the array name if it
refers to an array element), or if varName is unqualified (does
not include the names of any containing namespaces) but no
procedure is active, varName refers to a namespace variable
procedure is active, varName refers to a namespace variable
resolved according to the rules described under NAME RESOLUTION
in the namespace manual page.
If a procedure is active and varName is unqualified, then varName
refers to a parameter or local variable of the procedure, unless
varName was declared to resolve differently through one of the
varName was declared to resolve differently through one of the
global, variable, or upvar commands.
"
@values -min 1 -max 2
@ -924,7 +924,7 @@ tcl::namespace::eval punk::args::tclcore {
@cmd -name "builtin: tcl::string::cat" -help\
"Concatenate the given strings just like placing them directly next to each other and
return the resulting compound string. If no strings are present, the result is an
return the resulting compound string. If no strings are present, the result is an
empty string.
This primitive is occasionally handier than juxtaposition of strings when mixed quoting
is wanted, or when the aim is to return the result of a concatentation without resorting
@ -981,7 +981,7 @@ tcl::namespace::eval punk::args::tclcore {
in needleString. If found, return the index of the first character in the first such
match within haystackString. If there is no match, then return -1. If startIndex is
specified (in any of the forms described in STRING_INDICES), then the search is
constrained to start with the character in haystackString specified by the index.
constrained to start with the character in haystackString specified by the index.
"
@values -min 2 -max 3
needleString -type string
@ -1002,7 +1002,7 @@ tcl::namespace::eval punk::args::tclcore {
prepended to the string.
If index is at or after the end of the string (e.g., index is end), insertString is
appended to string."
@values -min 3 -max 3
string -type string
index -type indexexpression -help\
@ -1156,7 +1156,7 @@ tcl::namespace::eval punk::args::tclcore {
dict\
" Any proper dict structure,
with optional surrounding
whitespace. In case of
whitespace. In case of
improper dict structure, 0
is returned and the varname
will contain the index of
@ -1254,7 +1254,7 @@ tcl::namespace::eval punk::args::tclcore {
"If -strict is specified, then an empty string returns 0,
otherwise an empty string will return 1 on any class"
-failindex -type variablename -help\
"If -failindex is specified, then if the function returns 0,
"If -failindex is specified, then if the function returns 0,
the index in the string where the class was no longer valid will be stored
in the variable named."
@values -min 1 -max 1
@ -1329,7 +1329,7 @@ tcl::namespace::eval punk::args::tclcore {
" Invoke commandPrefix when the traced command is deleted.
Commands can be deleted explicitly using the rename command to
rename the command to an empty string. Commands are also deleted
when the interpreter is deleted, but traces will not be invoked
when the interpreter is deleted, but traces will not be invoked
because there is no interpreter in which to execute them."
}\
-help\
@ -1363,7 +1363,7 @@ tcl::namespace::eval punk::args::tclcore {
whenever command name is executed, with traces occurring at the points
indicated by the list ops. Name will be resolved using the usual namespace
resolution ruls used by commands. If the command does not exist, and error
will be thrown"
will be thrown"
name -type string -help\
"Name of command"
# ---------------------------------------------------------------
@ -1411,7 +1411,7 @@ tcl::namespace::eval punk::args::tclcore {
(the traced command for a enter operation, an arbitrary command
for an enterstep operation), including all arguments in their
fully expanded form. Op indicates what operation is being performed
on the command execution, and is on of enter or enterstep as
on the command execution, and is on of enter or enterstep as
defined above. The trace operation can be used to stop the command
from executing, by deleting the command in question. Of course when
the command is subsequently executed, an \"invalid command\" error
@ -1434,10 +1434,10 @@ tcl::namespace::eval punk::args::tclcore {
traces.
CommandPrefix executes in the same context as the code that invoked
the traced operation: thus the commandPrefix, if invoked from a
the traced operation: thus the commandPrefix, if invoked from a
procedure, will have access to the same local variables as code in the
procedure. This context may be different thatn the context in which
the trace was created. If commandPrefix invokes a procedure (which
the trace was created. If commandPrefix invokes a procedure (which
it normally does) then the procedure will have to use upvar or uplevel
commands if it wishes to access the local variables of the code which
invoked the trace operation.
@ -1463,13 +1463,13 @@ tcl::namespace::eval punk::args::tclcore {
@cmd -name "builtin: trace remove command" -help\
"If there is a trace set on command name with the operations and command
given by opList and commandPrefix, then the trace is removed, so that
commandPrefix will never again be invoked. Returns an empty string. If
commandPrefix will never again be invoked. Returns an empty string. If
name does not exist, the command will throw an error"
@values
name -type string -help\
"Name of command"
opList -type list -help\
"A list of one or more of the following items:
"A list of one or more of the following items:
rename
delete"
commandPrefix
@ -1483,25 +1483,25 @@ tcl::namespace::eval punk::args::tclcore {
"Create and initialise a namespace variable.
"
@form -form "setvalues" -synopsis "variable ?name value...? ?name?"
@values -min 2 -max -1
@values -min 2 -max -1
#todo
#In this case - we don't want name_value to display - as this is only used for documenting a builtin
#In this case - we don't want name_value to display - as this is only used for documenting a builtin
#For the case where an @argroups is used also for parsing - the help should display the synopsis form
#and also the name of the var in which it is placed.
# e.g
# ?{name value}...?
# ?{name value}...?
# (name_value)
#The second line giving an indication the resulting list of pairs can be accessed with something like:
# dict get $argd values name_value
#@arggroup -name name_value -min 1 -max 2 -optional 1 -multiple 1 -args {
# name
# name
# value
# }
@form -form "declare" -synopsis "variable name"
@values -min 1 -max 1
name -optional 0
@values -min 1 -max 1
name -optional 0
} "@doc -name Manpage: -url [manpage_tcl variable]"
# -- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- ---
@ -1514,7 +1514,7 @@ tcl::namespace::eval punk::args::tclcore {
}
}
punk::args::define {
@id -id ::zlib
@id -id ::zlib
@cmd -name "builtin: ::zlib" -help\
"zlib - compression and decompression operations
zlib version: ${$::punk::args::tclcore::argdoc::ZLIBVERSION}"
@ -1549,7 +1549,7 @@ tcl::namespace::eval punk::args::tclcore {
@id -id "::zlib adler32"
@cmd -name "builtin: ::zlib adler32" -help\
"Compute a checksum of binary string ${$I}string${$NI} using the Adler32
algorithm. If given, ${$I}initValue${$NI} is used to initialize the checksum engine.
algorithm. If given, ${$I}initValue${$NI} is used to initialize the checksum engine.
"
@values -min 1 -max 2
string -type string
@ -1561,7 +1561,7 @@ tcl::namespace::eval punk::args::tclcore {
#*** !doctools
#[subsection {Namespace punk::args::tclcore}]
#[para] Core API functions for punk::args::tclcore
#[para] Core API functions for punk::args::tclcore
#[list_begin definitions]
@ -1569,13 +1569,13 @@ tcl::namespace::eval punk::args::tclcore {
#proc sample1 {p1 n args} {
# #*** !doctools
# #[call [fun sample1] [arg p1] [arg n] [opt {option value...}]]
# #[para]Description of sample1
# #[para]Description of sample1
# #[para] Arguments:
# # [list_begin arguments]
# # [arg_def tring p1] A description of string argument p1.
# # [arg_def integer n] A description of integer argument n.
# # [list_end]
# return "ok"
# return "ok"
#}
@ -1595,14 +1595,14 @@ tcl::namespace::eval punk::args::tclcore::lib {
tcl::namespace::path [tcl::namespace::parent]
#*** !doctools
#[subsection {Namespace punk::args::tclcore::lib}]
#[para] Secondary functions that are part of the API
#[para] Secondary functions that are part of the API
#[list_begin definitions]
#proc utility1 {p1 args} {
# #*** !doctools
# #[call lib::[fun utility1] [arg p1] [opt {?option value...?}]]
# #[para]Description of utility1
# return 1
# #[para]Description of utility1
# return 1
#}
@ -1620,7 +1620,7 @@ tcl::namespace::eval punk::args::tclcore::lib {
#tcl::namespace::eval punk::args::tclcore::system {
#*** !doctools
#[subsection {Namespace punk::args::tclcore::system}]
#[para] Internal functions that are not part of the API
#[para] Internal functions that are not part of the API
@ -1632,11 +1632,11 @@ namespace eval ::punk::args::register {
lappend ::punk::args::register::NAMESPACES ::punk::args::tclcore ::punk::args::tclcore::argdoc
}
## Ready
## Ready
package provide punk::args::tclcore [tcl::namespace::eval punk::args::tclcore {
variable pkg punk::args::tclcore
variable version
set version 999999.0a1.0
set version 999999.0a1.0
}]
return

42
src/modules/punk/assertion-999999.0a1.0.tm

@ -21,7 +21,7 @@
#[manpage_begin punkshell_module_punk::assertion 0 999999.0a1.0]
#[copyright "2024"]
#[titledesc {assertion alternative to control::assert}] [comment {-- Name section and table of contents description --}]
#[moddesc {per-namespace assertions with }] [comment {-- Description at end of page heading --}]
#[moddesc {per-namespace assertions with }] [comment {-- Description at end of page heading --}]
#[require punk::assertion]
#[keywords module assertion assert debug]
#[description]
@ -99,9 +99,9 @@ tcl::namespace::eval punk::assertion::class {
}
# ++ +++ +++ +++ +++ +++ +++ +++ +++ +++ +++
#keep 2 namespaces for assertActive and assertInactive so there is introspection available via namespace origin
#keep 2 namespaces for assertActive and assertInactive so there is introspection available via namespace origin
tcl::namespace::eval punk::assertion::primary {
#tcl::namespace::export {[a-z]*}
#tcl::namespace::export {[a-z]*}
tcl::namespace::export assertActive assertInactive
proc assertActive {expr args} {
@ -112,7 +112,7 @@ tcl::namespace::eval punk::assertion::primary {
if {![tcl::string::is boolean -strict $res]} {
return -code error "invalid boolean expression: $expr"
}
if {$res} {return}
if {[llength $args]} {
@ -130,9 +130,9 @@ tcl::namespace::eval punk::assertion::primary {
}
tcl::namespace::eval punk::assertion::secondary {
tcl::namespace::export *
tcl::namespace::export *
#we need to actually define these procs here, (not import then re-export) - or namespace origin will report the original source namespace - which isn't what we want.
proc assertActive {expr args} [tcl::info::body ::punk::assertion::primary::assertActive]
proc assertActive {expr args} [tcl::info::body ::punk::assertion::primary::assertActive]
proc assertInactive args {}
}
@ -151,7 +151,7 @@ tcl::namespace::eval punk::assertion {
}
do_ns_import
#puts --------BBB
rename assertActive assert
rename assertActive assert
}
@ -162,20 +162,20 @@ tcl::namespace::eval punk::assertion {
#*** !doctools
#[subsection {Namespace punk::assertion}]
#[para] Core API functions for punk::assertion
#[para] Core API functions for punk::assertion
#[list_begin definitions]
#proc sample1 {p1 n args} {
# #*** !doctools
# #[call [fun sample1] [arg p1] [arg n] [opt {option value...}]]
# #[para]Description of sample1
# #[para]Description of sample1
# #[para] Arguments:
# # [list_begin arguments]
# # [arg_def tring p1] A description of string argument p1.
# # [arg_def integer n] A description of integer argument n.
# # [list_end]
# return "ok"
# return "ok"
#}
#like tcllib's control::assert - we are limited to the same callback for all namespaces.
@ -218,7 +218,7 @@ tcl::namespace::eval punk::assertion {
if {$on_off} {
#Enable it in calling namespace
if {"assert" eq $info_command} {
#There is an assert command reachable - due to namespace path etc, it could be in another namespace entirely - (not necessarily in an ancestor namespace of the namespace's tree structure)
#There is an assert command reachable - due to namespace path etc, it could be in another namespace entirely - (not necessarily in an ancestor namespace of the namespace's tree structure)
if {$which_assert eq [punk::assertion::system::nsjoin ${nscaller} assert]} {
tcl::namespace::eval $nscaller {
set assertorigin [tcl::namespace::origin assert]
@ -243,7 +243,7 @@ tcl::namespace::eval punk::assertion {
}
return 1
} else {
#assert is available, but isn't in the calling namespace - we should enable it in a way that is distinguishable from case where assert was explicitly imported to this namespace
#assert is available, but isn't in the calling namespace - we should enable it in a way that is distinguishable from case where assert was explicitly imported to this namespace
tcl::namespace::eval $nscaller {
set assertorigin [tcl::namespace::origin assert]
if {[tcl::string::match ::punk::assertion::* $assertorigin]} {
@ -303,8 +303,8 @@ tcl::namespace::eval punk::assertion {
return 0
}
} else {
#no assert command reachable
#If caller is using assert in this namespace - they should have imported it, or ensured it was reachable via namespace path
#no assert command reachable
#If caller is using assert in this namespace - they should have imported it, or ensured it was reachable via namespace path
puts stderr "no assert command visible from namespace '$nscaller' - use: namespace import ::punk::assertion::assert"
return 0
}
@ -327,14 +327,14 @@ tcl::namespace::eval punk::assertion::lib {
tcl::namespace::path [tcl::namespace::parent]
#*** !doctools
#[subsection {Namespace punk::assertion::lib}]
#[para] Secondary functions that are part of the API
#[para] Secondary functions that are part of the API
#[list_begin definitions]
#proc utility1 {p1 args} {
# #*** !doctools
# #[call lib::[fun utility1] [arg p1] [opt {?option value...?}]]
# #[para]Description of utility1
# return 1
# #[para]Description of utility1
# return 1
#}
@ -352,7 +352,7 @@ tcl::namespace::eval punk::assertion::lib {
tcl::namespace::eval punk::assertion::system {
#*** !doctools
#[subsection {Namespace punk::assertion::system}]
#[para] Internal functions that are not part of the API
#[para] Internal functions that are not part of the API
#Maintenance - snarfed from punk::ns to reduce dependencies - punk::ns::nsprefix is the master version
#nsprefix/nstail are string functions - they do not concern themselves with what namespaces are present in the system
@ -375,7 +375,7 @@ tcl::namespace::eval punk::assertion::system {
proc nstail {nspath args} {
#normalize the common case of ::::
set nspath [tcl::string::map [list :::: ::] $nspath]
set mapped [tcl::string::map [list :: \u0FFF] $nspath]
set mapped [tcl::string::map [list :: \u0FFF] $nspath]
set parts [split $mapped \u0FFF]
set defaults [list -strict 0]
@ -411,11 +411,11 @@ tcl::namespace::eval punk::assertion::system {
}
# ++ +++ +++ +++ +++ +++ +++ +++ +++ +++ +++
## Ready
## Ready
package provide punk::assertion [tcl::namespace::eval punk::assertion {
variable pkg punk::assertion
variable version
set version 999999.0a1.0
set version 999999.0a1.0
}]
return

200
src/modules/punk/basictelnet-999999.0a1.0.tm

@ -21,7 +21,7 @@
#[manpage_begin punkshell::basictelnet 0 999999.0a1.0]
#[copyright "2024"]
#[titledesc {basic telnet client - DKF/Wiki}] [comment {-- Name section and table of contents description --}]
#[moddesc {basic telnet client}] [comment {-- Description at end of page heading --}]
#[moddesc {basic telnet client}] [comment {-- Description at end of page heading --}]
#[require punk::basictelnet]
#[keywords module telnet protocol console terminal]
#[description]
@ -106,16 +106,16 @@ namespace eval punk::basictelnet {
#todo - use these as defaults - provide a way to configure/listen to local events and notify server (sigwinch unix, unknown windows)
set window_cols 80
set window_rows 25
set window_rows 25
#Some modern(?) telnet servers seem to just pump out utf-8 encoded graphics by default - without negotiating or confirming binary etc? review
variable encoding_guess utf-8
#we will experimentally assume utf-8 - which should handle ascii fine - and flip to cp437 when data encountered that cannot be valid utf-8
variable encoding_guess utf-8
#we will experimentally assume utf-8 - which should handle ascii fine - and flip to cp437 when data encountered that cannot be valid utf-8
#todo - proper charset negotiation
variable debug
set debug 0
set debug 0
proc debug {{on_off ""}} {
variable debug
if {$on_off eq ""} {
@ -125,9 +125,9 @@ namespace eval punk::basictelnet {
error "punk::basictelnet::debug on_off must be empty string to query, or a boolean value"
}
set debug [expr {$on_off}]
}
}
variable can_debug
variable can_debug
set can_debug 1
if {[catch {
package require textblock
@ -152,10 +152,10 @@ namespace eval punk::basictelnet {
#*** !doctools
#[subsection {Namespace punk::basictelnet}]
#[para] Core API functions for punk::basictelnet
#[para] Core API functions for punk::basictelnet
#[list_begin definitions]
variable optioncodes
variable optioncodes
dict set optioncodes 0 [list name "Binary Transmission" short "bin"]
dict set optioncodes 1 [list name "Echo" short "echo"]
dict set optioncodes 2 [list name "Reconnection" short "recon"]
@ -216,8 +216,8 @@ namespace eval punk::basictelnet {
dict set optioncodes 255 [list name "Extended-Options-List"]
#we are assuming we initiated the connection, and are in some sense the 'client'
variable server_option_state
variable client_option_state
variable server_option_state
variable client_option_state
variable client_option_declined
#not all these will make sense as a boolean? review.
#we use this also to support the Status option
@ -231,7 +231,7 @@ namespace eval punk::basictelnet {
set encoding_guess utf-8
dict for {k _v} $optioncodes {
dict set server_option_state $k 0 ;#DO from our perspective
dict set server_option_state $k 0 ;#DO from our perspective
dict set client_option_state $k 0 ;#WILL from our perspective
}
variable client_option_declined ;#record explicit negative responses (won'ts) to DO requests from server
@ -290,7 +290,7 @@ namespace eval punk::basictelnet {
# A rudimentary hardcoded configuration for options/negotiation
# The way in which features are enabled/disabled and what goes together needs refinement & better understanding
# todo - review
#Note: further logic required, for example even something as supposedly simple as echo shouldn't be active on both ends at once or we get a loop.
#Note: further logic required, for example even something as supposedly simple as echo shouldn't be active on both ends at once or we get a loop.
# Can't necessarily rely on other end not to allow us to do something insane.
# Probably also.. some options should be under direct user ability to initiate/control - not just a configuration
# For that to work fully we may need a separate punk::telnet package that has a pseudoterminal in front of the real console (scrolling sub-area), allowing a custom repl, custom status display etc.
@ -299,16 +299,16 @@ namespace eval punk::basictelnet {
#Passively enabled server features - ie those we don't initiate but will accept
#default response to WILL is WON'T
#define our positive responses here for those that we will do
variable respond_will_do
set respond_will_do [list]
variable respond_will_do
set respond_will_do [list]
lappend respond_will_do 0 ;#binary
lappend respond_will_do 1 ;#echo
lappend respond_will_do 3 ;#suppress go-ahead
lappend respond_will_do 5 ;#status - by agreeing to this we should be able to read unsolicited "IAC SB STATUS IS ... IAC SE" reports and compare to our perception of state. (and do something if mismatches?)
lappend respond_will_do 24 ;#remote is letting us know they are willing to send terminal-type - but we would still have to request it
#passively enabled client features - requests for our own behaviours we will respond positively
variable respond_do_will
#passively enabled client features - requests for our own behaviours we will respond positively
variable respond_do_will
set respond_do_will [list]
lappend respond_do_will 0 ;#binary
lappend respond_do_will 3 ;#Suppress go-ahead
@ -333,13 +333,13 @@ namespace eval punk::basictelnet {
#proc sample1 {p1 n args} {
# #*** !doctools
# #[call [fun sample1] [arg p1] [arg n] [opt {option value...}]]
# #[para]Description of sample1
# #[para]Description of sample1
# #[para] Arguments:
# # [list_begin arguments]
# # [arg_def tring p1] A description of string argument p1.
# # [arg_def integer n] A description of integer argument n.
# # [list_end]
# return "ok"
# return "ok"
#}
@ -402,28 +402,28 @@ namespace eval punk::basictelnet {
set client_declined "CLI-WONT:[a+ red bold][get_client_option_declined_summary][a]"
set info $server_summary\n$client_summary\n$client_declined\n$info
#set existing_handler [fileevent stdin readable]
set RST "\x1b\[m"
#set existing_handler [chan event stdin readable]
set RST "\x1b\[m"
set debug_width 80
set infoframe [textblock::frame -checkargs 0 -width $debug_width -ansiborder [a+ green bold] -title "[a cyan]Telnet Debug $terminal_type (encoding guess:$encoding_guess)$RST" $info]
#set w [textblock::width $infoframe]
set spacepatch "$RST[textblock::block $debug_width 4 { }]"
#puts -nonewline [punk::ansi::cursor_off]
#use non cursorsave version - slower - but less likely to interfere with cursor operations in data
#use non cursorsave version - slower - but less likely to interfere with cursor operations in data
set existing_input_handler [fileevent $inputchannel readable] ;#stdin
fileevent $inputchannel readable {}
set existing_input_handler [chan event $inputchannel readable] ;#stdin
chan event $inputchannel readable {}
if {[string length $outputchannel]} {
set existing_output_handler [fileevent $outputchannel readable] ;#sock
fileevent $outputchannel readable {}
set existing_output_handler [chan event $outputchannel readable] ;#sock
chan event $outputchannel readable {}
}
if {[catch {
if {[catch {
#90
set debug_offset [expr {$consolewidth - $debug_width}]
punk::console::move_emitblock_return 6 $debug_offset $spacepatch
punk::console::move_emitblock_return 6 $debug_offset $spacepatch
flush stdout
punk::console::move_emitblock_return 10 $debug_offset $infoframe
flush stdout
@ -432,11 +432,11 @@ namespace eval punk::basictelnet {
puts stderr "debug_frame error: $errM"
}
#todo - try? finally?
#todo - try? finally?
set writing_debug_frame 0
fileevent $inputchannel readable $existing_input_handler
chan event $inputchannel readable $existing_input_handler
if {[string length $outputchannel]} {
fileevent $outputchannel readable $existing_output_handler
chan event $outputchannel readable $existing_output_handler
}
return
}
@ -446,7 +446,7 @@ namespace eval punk::basictelnet {
variable debug
variable can_debug
variable debug_buffer
if {!$can_debug} {return}
if {!$can_debug} {return}
append debug_buffer $newlines
set lines [split $debug_buffer \n]
set lines [lrange $lines end-40 end]
@ -469,7 +469,7 @@ namespace eval punk::basictelnet {
server -type string -help\
"Hostname or IP address"
port -type integer -range {1 65535} -default 23 -help\
"TCP port"
"TCP port"
}
proc telnet {args} {
set argd [punk::args::get_by_id ::punk::basictelnet::telnet $args]
@ -513,11 +513,11 @@ namespace eval punk::basictelnet {
catch {set consolewidth [dict get [punk::console::get_size] columns]}
if {$consolewidth eq ""} {
#vt52?
set consolewidth 80
set consolewidth 80
}
if {$debug && $consolewidth-$::punk::basictelnet::window_cols < 80} {
puts stderr "Terminal width '$consolewidth' not wide enough for debug_window width: 80 + telnet window_cols:$::punk::basictelnet::window_cols"
puts stderr "Terminal width '$consolewidth' not wide enough for debug_window width: 80 + telnet window_cols:$::punk::basictelnet::window_cols"
puts stderr "Turn off debug, or make terminal window wider"
return
} elseif {$consolewidth < $::punk::basictelnet::window_cols} {
@ -525,17 +525,17 @@ namespace eval punk::basictelnet {
puts stderr "Ensure terminal is greater than or equal to punk::basictelnet::window_cols"
return
}
#todo - allow telnet with channels other than stdin/stdout - and multiple sessions - per session option_states
reset_option_states
set sock [socket $server $port]
#fconfigure $sock -buffering none -blocking 0 -encoding binary -translation crlf -eofchar {}
#fconfigure $sock -buffering none -blocking 0 -encoding binary -translation binary -eofchar {}
fconfigure $sock -buffering none -blocking 0 -encoding iso8859-1 -translation binary -eofchar {}
fconfigure stdout -buffering none
fileevent $sock readable [list [namespace current]::fromServer $sock]
#chan configure $sock -buffering none -blocking 0 -encoding binary -translation crlf -eofchar {}
#chan configure $sock -buffering none -blocking 0 -encoding binary -translation binary -eofchar {}
chan configure $sock -buffering none -blocking 0 -encoding iso8859-1 -translation binary -eofchar {}
chan configure stdout -buffering none
chan event $sock readable [list [namespace current]::fromServer $sock]
chan configure stdin -blocking 0
fileevent stdin readable [list [namespace current]::toServer $sock]
chan event stdin readable [list [namespace current]::toServer $sock]
variable closed
vwait ::punk::basictelnet::closed($sock)
unset closed($sock)
@ -563,12 +563,12 @@ namespace eval punk::basictelnet {
upvar ::punk::console::input_chunks_waiting input_chunks_waiting
set nextwaiting ""
if {[info exists input_chunks_waiting(stdin)] && [llength $input_chunks_waiting(stdin)]} {
set nextwaiting [lindex $input_chunks_waiting(stdin) 0]
if {[info exists input_chunks_waiting(stdin)] && [llength $input_chunks_waiting(stdin)]} {
set nextwaiting [lindex $input_chunks_waiting(stdin) 0]
set input_chunks_waiting(stdin) [lrange $input_chunks_waiting(stdin) 1 end]
}
fileevent stdin readable {}
chan event stdin readable {}
if {$nextwaiting eq ""} {
set chunk [read stdin]
} else {
@ -610,19 +610,19 @@ namespace eval punk::basictelnet {
puts stderr "Failed to write to socket $socket: data: [ansistring VIEW -lf 1 $chunk]"
set wrote_sock 0
}
if {$wrote_sock && ![eof $sock]} {
##################################################################################
#Re-enable channel read handler only if no waiting chunks - must process in order
##################################################################################
if {![llength $input_chunks_waiting(stdin)]} {
fileevent stdin readable [list [namespace current]::toServer $sock]
chan event stdin readable [list [namespace current]::toServer $sock]
} else {
#after idle [list [namespace current]::toServer $sock]
tailcall [namespace current]::toServer $sock
}
####################################################
#fileevent stdin readable [list [namespace current]::toServer $sock]
#chan event stdin readable [list [namespace current]::toServer $sock]
} else {
disconnect sock
}
@ -642,24 +642,24 @@ namespace eval punk::basictelnet {
variable encoding_guess
variable debug
variable fromserver_unprocessed
fileevent $sock readable {}
chan event $sock readable {}
variable in_sb
set chunksize 4096 ;#No choice of chunksize can avoid the possibility of splitting a token such as a Telnet protocol command or an ANSI sequence.
#in theory, a split ANSI sequence won't cause a problem - except if we have debug on which could emit a request on stdout (e.g get_cursor_pos)
#as a byte oriented supposedly ascii-by-default protocol - we shouldn't expect to get utf-8 without having negotiated it - but it looks suspiciously like this is the sort of thing that happens (2024) review? Examples? mapscii.me 1984.ws? Test.
#as a byte oriented supposedly ascii-by-default protocol - we shouldn't expect to get utf-8 without having negotiated it - but it looks suspiciously like this is the sort of thing that happens (2024) review? Examples? mapscii.me 1984.ws? Test.
#randomly chosen chunk boundaries - whether due to size or a combination of network speed and event scheduling can mean we get some utf8 characters split too.
set last_unprocessed $fromserver_unprocessed
set data $fromserver_unprocessed
set data $fromserver_unprocessed
set fromserver_unprocessed ""
append data [read $sock $chunksize]
#repeatedly appending when not fblocked - will somewhat reduce the risk of splitting both ANSI and TELNET commands - but at the cost of starving the output processing
#somewhat conveniently? - the IAC \xFF byte is not valid in utf-8 or ascii
#this whole mechanism may need to be reviewed/modified if/when Telnet binary mode and/or charset changing is implemented/understood by the author.
#this whole mechanism may need to be reviewed/modified if/when Telnet binary mode and/or charset changing is implemented/understood by the author.
#The current basic system is tested on the few available public telnet servers. - todo - test on some old industrial equipment, read more RFCs.
#for now we'll use punk::lib::get_utf8_leading as a hack way to determine if we should throw some trailing data aside for next loop to process?
#for now we'll use punk::lib::get_utf8_leading as a hack way to determine if we should throw some trailing data aside for next loop to process?
#while {![fblocked $sock] && ![eof $sock]} {
# add_debug "[a+ red bold]RE-READ[a]\n" stdin $sock
@ -685,16 +685,16 @@ namespace eval punk::basictelnet {
}
}
#mini debug buffer for each fromServer call - render using add_debug each loop
#mini debug buffer for each fromServer call - render using add_debug each loop
set debug_info ""
if {$debug} {
#only do this text-processing work if debug is on
append debug_info "------raw data [string length $data]---prev unprocessed:[string length $last_unprocessed]---" \n
#append debug_info [ansistring VIEW -lf 1 -vt 1 [encoding convertfrom utf-8 $data]] \n
#set rawview [ansistring VIEW -lf 1 -vt 1 [encoding convertfrom $encoding_guess $data]]
set rawview [ansistring VIEW -lf 1 -vt 1 $data]
#set viewblock [overtype::left -wrap 1 -width 78 -height 4 "" $rawview]
set viewblock [overtype::renderspace -cp437 1 -wrap 1 -width 78 -height 4 "" $rawview]
#set rawview [ansistring VIEW -lf 1 -vt 1 [encoding convertfrom $encoding_guess $data]]
set rawview [ansistring VIEW -lf 1 -vt 1 $data]
#set viewblock [overtype::left -wrap 1 -width 78 -height 4 "" $rawview]
set viewblock [overtype::renderspace -cp437 1 -wrap 1 -width 78 -height 4 "" $rawview]
set lines [split $viewblock \n]
if {[llength $lines] > 4} {
append debug_info [join [list {*}[lrange $lines 0 1] "...<[expr {[llength $lines] -4}] lines undisplayed>..." {*}[lrange $lines end-1 end]] \n]
@ -711,7 +711,7 @@ namespace eval punk::basictelnet {
#---------------
#TODO - fix possible chunk boundary that gives us an incomplete IAC sequence.
#As it stands - we won't properly handle it - possible it will cause intermittent telnet protocol bugs!
#will need a mechanism within protocol function and loop to abort and throw back to next fromServer event
#will need a mechanism within protocol function and loop to abort and throw back to next fromServer event
#---------------
while 1 {
if {!$in_sb} {
@ -732,20 +732,20 @@ namespace eval punk::basictelnet {
if {$post_IAC_byte < "\xef"} {
#??
#write \xf0$post_IAC_byte ;#from wiki code. purpose not understood.
puts stderr "unexpected - byte less than EF following IAC"
puts stderr "unexpected - byte less than EF following IAC"
set data [string range $data $idx+1 end]
incr idx
} elseif {$post_IAC_byte == "\xff"} {
#write \xf0 ;#?? This came from wiki code - intention unclear.. latin small letter Eth
#RFC indicates double up of \xff is treated as literal
#this can't be part of utf-8 -
#this can't be part of utf-8 -
puts -nonewline stdout \xff
set data [string range $data $idx+2 end]
incr idx 2
} else {
incr idx 2
set ophex ""
#telnet commands are at least 2 bytes
#telnet commands are at least 2 bytes
binary scan $post_IAC_byte H2 cmdhex
switch -- $cmdhex {
fb - fc - fd - fe {
@ -773,7 +773,7 @@ namespace eval punk::basictelnet {
binary scan $opbyte H2 ophex
}
default {
}
}
protocol $sock $cmdhex $ophex
@ -789,7 +789,7 @@ namespace eval punk::basictelnet {
ff {
#expecting SE next - but will pass to protocol as if it's the 'cmd' for handling/verification
set expectedSE [string index $data 1]
binary scan $expectedSE H2 expectedSEhex
binary scan $expectedSE H2 expectedSEhex
protocol $sock $expectedSEhex ""
}
default {
@ -826,14 +826,14 @@ namespace eval punk::basictelnet {
puts -nonewline stdout [encoding convertfrom $encoding_guess $prefix]
} else {
set fromserver_unprocessed ""
#look for incomplete ansi sequences
#REVIEW - encoding ?
set ansisplits [punk::ansi::ta::split_codes_single $prefix]
set last_pt [lindex $ansisplits end] ;#last part is supposed to be plaintext - if it looks like it contains a partial ansi - throw it to fromserver_unprocessed for next fromServer call
#look for incomplete ansi sequences
#REVIEW - encoding ?
set ansisplits [punk::ansi::ta::split_codes_single $prefix]
set last_pt [lindex $ansisplits end] ;#last part is supposed to be plaintext - if it looks like it contains a partial ansi - throw it to fromserver_unprocessed for next fromServer call
if {[string first "\x1b" $last_pt] >= 0} {
set complete [join [lrange $ansisplits 0 end-1] ""]
puts -nonewline stdout [encoding convertfrom $encoding_guess $complete]
set fromserver_unprocessed $last_pt
set fromserver_unprocessed $last_pt
} else {
puts -nonewline stdout [encoding convertfrom $encoding_guess $prefix]
}
@ -872,24 +872,24 @@ namespace eval punk::basictelnet {
#after idle [list fileevent $sock readable [list [namespace current]::fromServer $sock]]
#after idle [list chan event $sock readable [list [namespace current]::fromServer $sock]]
if {[string length $fromserver_unprocessed]} {
#review - by throwing to another loop without waiting for readable event - we could spin on same data...?
#after idle [list [namespace current]::fromServer $sock]
fileevent $sock readable [list [namespace current]::fromServer $sock]
chan event $sock readable [list [namespace current]::fromServer $sock]
} else {
fileevent $sock readable [list [namespace current]::fromServer $sock]
chan event $sock readable [list [namespace current]::fromServer $sock]
}
}
proc disconnect {sock} {
variable closed
puts stdout "local disconnect"
catch {fileevent $sock readable {}}
catch {chan event $sock readable {}}
catch {close $sock}
set closed($sock) 1
fileevent stdin readable {}
chan event stdin readable {}
}
proc write string {
@ -927,12 +927,12 @@ namespace eval punk::basictelnet {
}
proc protocol {sock cmdhex ophex} {
variable in_sb
variable sb_state
variable optioncodes
variable sb_state
variable optioncodes
variable respond_will_do
variable respond_do_will
variable client_option_state ;#WILLs
variable client_option_declined ;#WON'Ts - but only those that were actually requested by server - not our default won'ts
variable client_option_declined ;#WON'Ts - but only those that were actually requested by server - not our default won'ts
variable server_option_state ;#DOs
upvar 1 debug_info debug_info
@ -952,7 +952,7 @@ namespace eval punk::basictelnet {
}
flush stderr
switch $cmdhex {
f0 {# SE - End of subnegoatiation parameters 240
f0 {# SE - End of subnegoatiation parameters 240
#error to get when not in sb?
puts stderr "Unexpected SE. We don't appear to be in SB!"
flush stderr
@ -975,7 +975,7 @@ namespace eval punk::basictelnet {
flush $sock
}
f7 {# EC - Erase Character 247
write \u007f
write \u007f
}
f8 {# EL - Erase Line 248
write \u0019
@ -989,7 +989,7 @@ namespace eval punk::basictelnet {
if {[dict get $client_option_state $opdec] || [dict get $server_option_state $opdec]} {
incr idx
#action for many subnegotiations is SEND=1 or IS=0
set actionbyte [string index $data $idx]
set actionbyte [string index $data $idx]
set actiondec [scan $actionbyte %c]
incr idx ;#for action
switch -- $opdec {
@ -998,8 +998,8 @@ namespace eval punk::basictelnet {
switch -- $actiondec {
0 {
#IS
#we should only get these reports if status is in our DO list
#keep in_sb as 1 and initialise sb_state
#we should only get these reports if status is in our DO list
#keep in_sb as 1 and initialise sb_state
dict set sb_state opdec $opdec
dict set sb_state actiondec 0
dict set sb_state data [dict create]
@ -1009,7 +1009,7 @@ namespace eval punk::basictelnet {
#we should only get a request to send status if it is in our WILL list
#expect the IAC SE to immediately follow
if {[string range $data $idx $idx+1] ne "\xff\xf0"} {
error "malformed send status request"
error "malformed send status request"
}
incr idx 2
if {![dict get $client_option_state $opdec]} {
@ -1054,8 +1054,8 @@ namespace eval punk::basictelnet {
switch -- $actiondec {
0 {
#IS
#we should only get these reports if status is in our DO list
#as maximum
#we should only get these reports if status is in our DO list
#as maximum
set nextSE [string first \xff\xf0 $data]
if {$nextSE > 0} {
set remote_terminal_type [string range $data $idx $nextSE-1]
@ -1064,7 +1064,7 @@ namespace eval punk::basictelnet {
#could presumably happen.. todo
error "didn't receive terminal-type in single chunk - review code"
}
##keep in_sb as 1 and initialise sb_state
##keep in_sb as 1 and initialise sb_state
#dict set sb_state opdec $opdec
#dict set sb_state actiondec 0
#dict set sb_state data [dict create]
@ -1074,7 +1074,7 @@ namespace eval punk::basictelnet {
#we should only get a request to send status if it is in our WILL list
#expect the IAC SE to immediately follow
if {[string range $data $idx $idx+1] ne "\xff\xf0"} {
error "malformed send status request"
error "malformed send status request"
}
incr idx 2
if {![dict get $client_option_state $opdec]} {
@ -1099,7 +1099,7 @@ namespace eval punk::basictelnet {
}
default {
#if we've responded positively to supporting the option - it should have a switch-arm here
error "No switch handler for option '$opdec' [dict get $optioncodes $opdec]"
error "No switch handler for option '$opdec' [dict get $optioncodes $opdec]"
}
}
} else {
@ -1108,7 +1108,7 @@ namespace eval punk::basictelnet {
#todo - ignore?
#we shouldn't get here if we are properly in sync with a well-behaved partner
#if we do however.. we need to either abort immediately.. or ignore the subnegotiation by skipping ahead to SE as it may not even be an SB structure we understand.
#let's try the ignore option first..
#let's try the ignore option first..
set next_SE [string first \xff\xf0 $data]
if {$next_SE >=0} {
set idx [expr {$next_SE +2}]
@ -1123,9 +1123,9 @@ namespace eval punk::basictelnet {
variable respond_will_do
set byte [string index $data $idx]
if {$opdec in $respond_will_do} {
if {[dict get $server_option_state $opdec]} {
#already known DO
} else {
if {[dict get $server_option_state $opdec]} {
#already known DO
} else {
append debug_info ">>>responding to server WILL declaration. DO $opdec [dict get $optioncodes $opdec]<<<" \n
puts -nonewline $sock \xff\xfd$byte ;#respond DO
dict set server_option_state $opdec 1
@ -1201,7 +1201,7 @@ namespace eval punk::basictelnet {
dict for {opt state} $server_option_state {
if {$state} {
if {![dict exists $reported_state will $opt]} {
lappend mismatches [list server $opt reported DON'T stored DO]
lappend mismatches [list server $opt reported DON'T stored DO]
}
} else {
if {[dict exists $reported_state will $opt]} {
@ -1251,7 +1251,7 @@ namespace eval punk::basictelnet {
dict set sb_state data $existing_data ;#updated
}
}
}
}
}
}
@ -1271,14 +1271,14 @@ namespace eval punk::basictelnet::lib {
namespace path [namespace parent]
#*** !doctools
#[subsection {Namespace punk::basictelnet::lib}]
#[para] Secondary functions that are part of the API
#[para] Secondary functions that are part of the API
#[list_begin definitions]
#proc utility1 {p1 args} {
# #*** !doctools
# #[call lib::[fun utility1] [arg p1] [opt {?option value...?}]]
# #[para]Description of utility1
# return 1
# #[para]Description of utility1
# return 1
#}
@ -1296,17 +1296,17 @@ namespace eval punk::basictelnet::lib {
namespace eval punk::basictelnet::system {
#*** !doctools
#[subsection {Namespace punk::basictelnet::system}]
#[para] Internal functions that are not part of the API
#[para] Internal functions that are not part of the API
}
# ++ +++ +++ +++ +++ +++ +++ +++ +++ +++ +++
## Ready
## Ready
package provide punk::basictelnet [namespace eval punk::basictelnet {
variable pkg punk::basictelnet
variable version
set version 999999.0a1.0
set version 999999.0a1.0
}]
return

60
src/modules/punk/cap-999999.0a1.0.tm

@ -26,7 +26,7 @@
#[para]punk::cap provides management of named capabilities and the provider packages and handler packages that implement a pluggable capability.
#[para]see also [uri https://core.tcl-lang.org/tcllib/doc/trunk/embedded/md/tcllib/files/modules/pluginmgr/pluginmgr.md {tcllib pluginmgr}] for an alternative which uses safe interpreters
#[subsection Concepts]
#[para]A [term capability] may be something like providing a folder of files, or just a data dictionary, and/or an API
#[para]A [term capability] may be something like providing a folder of files, or just a data dictionary, and/or an API
#
#[para][term {capability handler}] - a package/namespace which may provide validation and standardised ways of looking up provider data
# registered (or not) using register_capabilityname <capname> <capnamespace>
@ -49,7 +49,7 @@ package require oolib
# ++ +++ +++ +++ +++ +++ +++ +++ +++ +++ +++
tcl::namespace::eval punk::cap {
variable pkgcapsdeclared [tcl::dict::create]
variable pkgcapsdeclared [tcl::dict::create]
variable pkgcapsaccepted [tcl::dict::create]
variable caps [tcl::dict::create]
namespace eval class {
@ -71,8 +71,8 @@ tcl::namespace::eval punk::cap {
#*** !doctools
#[call class::interface_caphandler.registry [method pkg_register] [arg pkg] [arg capname] [arg capdict] [arg fullcapabilitylist]]
#handler may override and return 0 (indicating don't register)e.g if pkg capdict data wasn't valid
#overridden handler must be able to handle multiple calls for same pkg - but it may return 1 or 0 as it wishes.
return 1 ;#default to permit
#overridden handler must be able to handle multiple calls for same pkg - but it may return 1 or 0 as it wishes.
return 1 ;#default to permit
}
method pkg_unregister {pkg} {
#*** !doctools
@ -106,9 +106,9 @@ tcl::namespace::eval punk::cap {
oo::class create ::punk::cap::class::interface_capprovider.registration {
#*** !doctools
# [enum] CLASS [class interface_cappprovider.registration]
# [para]Your provider package will need to instantiate this object under a sub-namespace called [namespace capsystem] within your package namespace.
# [para]Your provider package will need to instantiate this object under a sub-namespace called [namespace capsystem] within your package namespace.
# [para]If your package namespace is mypackages::providerpkg then the object command would be at mypackages::providerpkg::capsystem::capprovider.registration
# [para]Example code for your provider package to evaluate within its namespace:
# [para]Example code for your provider package to evaluate within its namespace:
# [example {
#namespace eval capsystem {
# if {[info commands capprovider.registration] eq ""} {
@ -133,7 +133,7 @@ tcl::namespace::eval punk::cap {
#[para] This method must be overridden by your provider using oo::objdefine cappprovider.registration as in the example above.
# There must be at least one 2-element list in the result for the provider to be registerable.
#[para]The first element of the list is the capabilityname - which can be custom to your provider/handler packages - or a well-known name that other authors may use/implement.
#[para]The second element is a dictionary of keys specific to the capability being implemented. It may be empty if the any potential capability handlers for the named capability don't require registration data.
#[para]The second element is a dictionary of keys specific to the capability being implemented. It may be empty if the any potential capability handlers for the named capability don't require registration data.
error "interface_capprovider.registration not implemented by provider"
}
#*** !doctools
@ -142,11 +142,11 @@ tcl::namespace::eval punk::cap {
oo::class create ::punk::cap::class::interface_capprovider.provider {
#*** !doctools
# [enum] CLASS [class interface_capprovider.provider]
# [para] Your provider package will need to instantiate this directly under it's own namespace with the command name of [emph {provider}]
# [enum] CLASS [class interface_capprovider.provider]
# [para] Your provider package will need to instantiate this directly under it's own namespace with the command name of [emph {provider}]
# [example {
# namespace eval mypackages::providerpkg {
# punk::cap::class::interface_capprovider.provider create provider mypackages::providerpkg
# namespace eval mypackages::providerpkg {
# punk::cap::class::interface_capprovider.provider create provider mypackages::providerpkg
# }
# }]
# [list_begin definitions]
@ -229,7 +229,7 @@ tcl::namespace::eval punk::cap {
#Not all capability names have to be registered.
#A package registering as a provider using register_package can include capabilitynames in it's capabilitylist which have no associated handler.
#such unregistered capabilitynames may be used just to flag something, or have datamembers significant to callers cooperatively interested in that capname.
#such unregistered capabilitynames may be used just to flag something, or have datamembers significant to callers cooperatively interested in that capname.
#we allow registering a capability with an empty handler (capnamespace) - but this means another handler could be registered later.
proc register_capabilityname {capname capnamespace} {
#puts stderr "REGISTER_CAPABILITYNAME $capname $capnamespace"
@ -243,10 +243,10 @@ tcl::namespace::eval punk::cap {
}
}
#allow register of existing capname iff there is no current handler
#as handlers can be used to validate during provider registration - ideally handlers should be registered before any pkgs call register_package
#we allow loading a handler later though - but will need to validate existing data from pkgs that have already registered as providers
#as handlers can be used to validate during provider registration - ideally handlers should be registered before any pkgs call register_package
#we allow loading a handler later though - but will need to validate existing data from pkgs that have already registered as providers
if {[set hdlr [capability_get_handler $capname]] ne ""} {
puts stderr "register_capabilityname cannot register capability:$capname with handler:$capnamespace. There is already a registered handler:$hdlr"
puts stderr "register_capabilityname cannot register capability:$capname with handler:$capnamespace. There is already a registered handler:$hdlr"
return
}
#assertion: capnamespace may or may not be empty string, capname may or may not already exist in caps dict, caps $capname providers may have existing entries.
@ -295,14 +295,14 @@ tcl::namespace::eval punk::cap {
if {$count == 0} {
set pkgposn [lsearch $providers $pkg]
if {$pkgposn >= 0} {
set updated_providers [lreplace $providers $posn $posn]
set updated_providers [lreplace $providers $posn $posn]
tcl::dict::set caps $capname providers $updated_providers
}
}
}
}
}
}
proc capability_exists {capname} {
@ -328,7 +328,7 @@ tcl::namespace::eval punk::cap {
if {[tcl::dict::exists $caps $capname]} {
return [tcl::dict::get $caps $capname handler]
}
return ""
return ""
}
proc call_handler {capname args} {
if {[set handler [capability_get_handler $capname]] eq ""} {
@ -461,7 +461,7 @@ tcl::namespace::eval punk::cap {
#todo!
proc unregister_package {pkg {capname *}} {
variable pkgcapsdeclared
variable pkgcapsdeclared
variable caps
if {[string match ::* $pkg]} {
set pkg [string range $pkg 2 end]
@ -471,7 +471,7 @@ tcl::namespace::eval punk::cap {
set capabilitylist [dict get $pkgcapsdeclared $pkg]
foreach c $capabilitylist {
set do_unregister 1
lassign $c capname _capdict
lassign $c capname _capdict
set cap_info [dict get $caps $capname]
set pkglist [dict get $cap_info providers]
set posn [lsearch $pkglist $pkg]
@ -479,9 +479,9 @@ tcl::namespace::eval punk::cap {
if {[set capreg [punk::cap::capsystem::get_caphandler_registry $capname]] ne ""} {
#review
# it seems not useful to allow the callback to block this unregister action
#the pkg may have multiple datasets for each capname so callback will only be called for first dataset we encounter
#vetoing unregister would make this more complex for no particular advantage
#if per dataset deregistration required this should probably be a separate thing
#the pkg may have multiple datasets for each capname so callback will only be called for first dataset we encounter
#vetoing unregister would make this more complex for no particular advantage
#if per dataset deregistration required this should probably be a separate thing
$capreg pkg_unregister $pkg $capname
}
set pkglist [lreplace $pkglist $posn $posn]
@ -510,7 +510,7 @@ tcl::namespace::eval punk::cap {
}
}
proc pkgcaps {} {
variable pkgcapsdeclared
variable pkgcapsdeclared
variable pkgcapsaccepted
set result [dict create]
foreach {pkg capsdeclared} $pkgcapsdeclared {
@ -522,7 +522,7 @@ tcl::namespace::eval punk::cap {
dict set result $pkg accepted $accepted
}
return $result
}
}
proc capability {capname} {
variable caps
@ -565,14 +565,14 @@ tcl::namespace::eval punk::cap {
#[subsection {Namespace punk::cap::advanced}]
#[para] punk::cap::advanced API. Functions here are generally not the preferred way to interact with punk::cap.
#[para] In some cases they may allow interaction in less safe ways or may allow use of features that are unavailable in the base namespace.
#[para] Some functions are here because they are only marginally or rarely useful, and they are here to keep the base API simple.
#[para] Some functions are here because they are only marginally or rarely useful, and they are here to keep the base API simple.
#[list_begin definitions]
proc promote_provider {pkg} {
#*** !doctools
# [call advanced::[fun promote_provider] [arg pkg]]
#[para]Move the named provider package to the preferred end of the list (tail).
#[para]The active handler may or may not utilise this for preferencing. See documentation for the specific handler package to confirm.
#[para]The active handler may or may not utilise this for preferencing. See documentation for the specific handler package to confirm.
#[para]
#[para] promote/demote doesn't always make a lot of sense .. should preferably be configurable per capapbility for multicap provider pkgs
#[para]The idea is to provide a crude way to preference/depreference packages independently of order the packages were loaded
@ -615,7 +615,7 @@ tcl::namespace::eval punk::cap {
#*** !doctools
# [call advanced::[fun demote_provider] [arg pkg]]
#[para]Move the named provider package to the preferred end of the list (tail).
#[para]The active handler may or may not utilise this for preferencing. See documentation for the specific handler package to confirm.
#[para]The active handler may or may not utilise this for preferencing. See documentation for the specific handler package to confirm.
variable pkgcapsdeclared
variable caps
if {[string match ::* $pkg]} {
@ -677,11 +677,11 @@ tcl::namespace::eval punk::cap {
# ++ +++ +++ +++ +++ +++ +++ +++ +++ +++ +++
## Ready
## Ready
package provide punk::cap [namespace eval punk::cap {
variable version
variable pkg punk::cap
set version 999999.0a1.0
set version 999999.0a1.0
variable README.md [string map [list %pkg% $pkg %ver% $version] {
# punk capabilities system
## pkg: %pkg% version: %ver%

4
src/modules/punk/cap/handlers/caphandler-999999.0a1.0.tm

@ -43,10 +43,10 @@ namespace eval punk::cap::handlers::caphandler {
# ++ +++ +++ +++ +++ +++ +++ +++ +++ +++ +++
## Ready
## Ready
package provide punk::cap::handlers::caphandler [namespace eval punk::cap::handlers::caphandler {
variable pkg punk::cap::handlers::caphandler
variable version
set version 999999.0a1.0
set version 999999.0a1.0
}]
return

78
src/modules/punk/cap/handlers/templates-999999.0a1.0.tm

@ -23,7 +23,7 @@ package require punk::repo
# ++ +++ +++ +++ +++ +++ +++ +++ +++ +++ +++
#register using:
# punk::cap::register_capabilityname templates ::punk::cap::handlers::templates
# punk::cap::register_capabilityname templates ::punk::cap::handlers::templates
#By convention and for consistency, we don't register here during package loading - but require the calling app to do it.
# (even if it tends to be done immediately after package require anyway)
@ -67,11 +67,11 @@ namespace eval punk::cap::handlers::templates {
#for template pathtype module & shellproject* we can resolve whether it's within a project at registration time and store the projectbase rather than rechecking it each time the templates handler api is called
#for template pathtype absolute - we can do the same.
#There is a small chance for a long-running shell that a project is later created which makes the absolute path within a project - but it seems an unlikely case, and probably won't surprise the user that they need to relaunch the shell or reload the capsystem to see the change.
#for template pathtype absolute - we can do the same.
#There is a small chance for a long-running shell that a project is later created which makes the absolute path within a project - but it seems an unlikely case, and probably won't surprise the user that they need to relaunch the shell or reload the capsystem to see the change.
#adhoc and currentproject* paths are relative to cwd - so no projectbase information can be stored at registration time.
#not all template item types will need projectbase information - as the item data may be self-contained within the template structure -
#not all template item types will need projectbase information - as the item data may be self-contained within the template structure -
#but project_layout will need it - or at least need to know if there is no project - because project_layout data is never stored in the template folder structure directly.
switch -- $pathtype {
adhoc {
@ -95,7 +95,7 @@ namespace eval punk::cap::handlers::templates {
} else {
set tm_exists [file exists $tmfile]
}
if {![file exists $tmfile]} {
if {!$tm_exists} {
puts stderr "punk::cap::handlers::templates::capsystem::pkg_register WARNING - unable to determine base folder for package '$pkg' which is attempting to register with punk::cap as a provider of '$capname' capability"
flush stderr
return 0
@ -128,7 +128,7 @@ namespace eval punk::cap::handlers::templates {
}
set extended_capdict $capdict
dict set extended_capdict vendor $vendor ;#vendor key still required.. controlling vendor?
dict set extended_capdict vendor $vendor ;#vendor key still required.. controlling vendor?
}
currentproject {
if {[file pathtype $path] ne "relative"} {
@ -140,7 +140,7 @@ namespace eval punk::cap::handlers::templates {
set extended_capdict $capdict
dict set extended_capdict vendor $vendor
dict set extended_capdict vendor $vendor
}
shellproject {
if {[file pathtype $path] ne "relative"} {
@ -150,7 +150,7 @@ namespace eval punk::cap::handlers::templates {
set shellbase [file dirname [file dirname [file normalize [set ::argv0]/__]]] ;#review
set projectinfo [punk::repo::find_repos $shellbase]
set projectbase [dict get $projectinfo closest]
set extended_capdict $capdict
dict set extended_capdict vendor $vendor
dict set extended_capdict projectbase $projectbase
@ -170,7 +170,7 @@ namespace eval punk::cap::handlers::templates {
set projectbase [dict get $projectinfo closest]
set extended_capdict $capdict
dict set extended_capdict vendor $vendor
dict set extended_capdict vendor $vendor
dict set extended_capdict projectbase $projectbase
}
absolute {
@ -188,7 +188,7 @@ namespace eval punk::cap::handlers::templates {
#todo - verify no other provider has registered same absolute path - if sharing a project-external location is needed - they need their own subfolder
set extended_capdict $capdict
dict set extended_capdict resolved_path $normpath
dict set extended_capdict resolved_path $normpath
dict set extended_capdict vendor $vendor
dict set extended_capdict projectbase $projectbase
}
@ -199,7 +199,7 @@ namespace eval punk::cap::handlers::templates {
}
# -- --- --- --- --- --- --- ---- ---
# update package internal data
# update package internal data
# -- --- --- --- --- --- --- ---- ---
upvar ::punk::cap::handlers::templates::provider_info_$cname provider_info
@ -208,13 +208,13 @@ namespace eval punk::cap::handlers::templates {
}
if {![info exists provider_info] || $extended_capdict ni [dict get $provider_info $pkg]} {
#this checks for duplicates from the same provider - but not if other providers already added the path
#review -
#review -
dict lappend provider_info $pkg $extended_capdict
}
# -- --- --- --- --- --- --- ---- ---
# instantiation of api at punk::cap::handlers::templates::api_$capname
# instantiation of api at punk::cap::handlers::templates::api_$capname
# -- --- --- --- --- --- --- ---- ---
set apicmd "::punk::cap::handlers::templates::api_$capname"
if {[info commands $apicmd] eq ""} {
@ -227,12 +227,12 @@ namespace eval punk::cap::handlers::templates {
upvar ::punk::cap::handlers::templates::handled_caps hcaps
foreach capname $hcaps {
set cname [string map {. _} $capname]
upvar ::punk::cap::handlers::templates::provider_info_$cname my_provider_info
upvar ::punk::cap::handlers::templates::provider_info_$cname my_provider_info
dict unset my_provider_info $pkg
#destroy api objects?
}
}
}
}
}
}
@ -293,7 +293,7 @@ namespace eval punk::cap::handlers::templates {
set found_paths_absolute [list]
foreach pkg $providerpkg {
foreach pkg $providerpkg {
set found_paths [list]
#set acceptedlist [dict get [punk::cap::pkgcap $pkg $capabilityname] accepted]
@ -314,13 +314,13 @@ namespace eval punk::cap::handlers::templates {
set module_projectroot [dict get $capdecl_extended projectbase]
dict lappend found_paths_module $vendor [list pkg $pkg path [dict get $capdecl_extended resolved_path] pathtype $pathtype projectbase $module_projectroot]
} elseif {$pathtype eq "currentproject_multivendor"} {
set searchbase $startdir
set searchbase $startdir
set pathinfo [punk::repo::find_repos $searchbase]
set pwd_projectroot [dict get $pathinfo closest]
if {$pwd_projectroot ne ""} {
set deckbase [file join $pwd_projectroot $path]
if {![file exists $deckbase]} {
continue
continue
}
#add vendor/x folders first - earlier in list is lower priority
set vendorbase [file join $deckbase vendor]
@ -349,7 +349,7 @@ namespace eval punk::cap::handlers::templates {
}
}
} elseif {$pathtype eq "currentproject"} {
set searchbase $startdir
set searchbase $startdir
set pathinfo [punk::repo::find_repos $searchbase]
set pwd_projectroot [dict get $pathinfo closest]
if {$pwd_projectroot ne ""} {
@ -369,7 +369,7 @@ namespace eval punk::cap::handlers::templates {
if {$shell_projectroot ne ""} {
set deckbase [file join $shell_projectroot $path]
if {![file exists $deckbase]} {
continue
continue
}
#add vendor/x folders first - earlier in list is lower priority
set vendorbase [file join $deckbase vendor]
@ -471,19 +471,19 @@ namespace eval punk::cap::handlers::templates {
return $folderdict
}
method get_itemdict_projectlayouts {args} {
set argd [punk::args::get_dict {
set argd [punk::args::get_dict {
@id -id "::punk::cap::handlers::templates::class::api get_itemdict_projectlayouts"
@opts -anyopts 1
#peek -startdir while allowing all other opts/vals to be verified down-the-line instead of here
-startdir -default ""
@values -maxvalues -1
} $args]
} $args]
set opt_startdir [dict get $argd opts -startdir]
if {$opt_startdir eq ""} {
set searchbase [pwd]
} else {
set searchbase $opt_startdir
set searchbase $opt_startdir
}
set refdict [my get_itemdict_projectlayoutrefs {*}$args]
@ -502,7 +502,7 @@ namespace eval punk::cap::handlers::templates {
# e.g ref may be @vendor+punks+othersample@sample-0.1 or layoutalias-1.1@vendor+punk+othersample@sample-0.1
#there must always be an @ before vendor or custom . There is either a template-name alias or empty string before this first @
#trim off first @ part
set tailats [join [lrange $atparts 1 end] @]
set tailats [join [lrange $atparts 1 end] @]
# @ parts after the first are part of the path within the project_layouts structure
set subpathlist [split $tailats +]
if {[dict exists $refinfo sourceinfo projectbase]} {
@ -553,7 +553,7 @@ namespace eval punk::cap::handlers::templates {
if {$vendor ne "_project"} {
set itemname $vendor.$itemname
}
return $itemname
return $itemname
}}}
}
set arglist [concat $config $args]
@ -623,7 +623,7 @@ namespace eval punk::cap::handlers::templates {
}}}\
-command_get_item_name {apply {{vendor basefolder itempath} {
set relativepath [punk::path::relative $basefolder $itempath]
set relativepath [punk::path::relative $basefolder $itempath]
set dirs [file dirname $relativepath]
if {$dirs eq "."} {
set dirs ""
@ -636,7 +636,7 @@ namespace eval punk::cap::handlers::templates {
}
if {$vendor ne "_project"} {
set tname ${vendor}.$tname
}
}
return $tname
}}}
}
@ -645,11 +645,11 @@ namespace eval punk::cap::handlers::templates {
}
#shared algorithm for get_itemdict_* methods
#requires a -templatefolder_subdir indicating a directory within each template base folder in which to search
#requires a -templatefolder_subdir indicating a directory within each template base folder in which to search
#and a file selection mechanism command -command_get_items_from_base
#and a name determining command -command_get_item_name
method _get_itemdict {args} {
set argd [punk::args::get_dict {
set argd [punk::args::get_dict {
@id -id "::punk::cap::handlers::templates::class::api _get_itemdict"
@cmd -name _get_itemdict
@opts -anyopts 0
@ -657,7 +657,7 @@ namespace eval punk::cap::handlers::templates {
-templatefolder_subdir -optional 0
-command_get_items_from_base -optional 0
-command_get_item_name -optional 0
-not -default "" -multiple 1
-not -default "" -multiple 1
@values -maxvalues -1
globsearches -default * -multiple 1
} $args]
@ -697,12 +697,12 @@ namespace eval punk::cap::handlers::templates {
set items_here [dict create] ;#maintain a list keyed on name for sorting within this base only
foreach itempath $matches {
set itemname [{*}$opt_command_get_item_name $vendor $basefolder $itempath]
dict set items_here $itemname [list item $itempath baseinfo $baseinfo]
dict set items_here $itemname [list item $itempath baseinfo $baseinfo]
#lappend items [list item $itempath baseinfo $baseinfo]
}
set ordered_names [lsort [dict keys $items_here]]
#add to the outer items list
foreach nm $ordered_names {
#add to the outer items list
foreach nm $ordered_names {
set iteminfo [dict get $items_here $nm]
lappend items [list originalname $nm iteminfo $iteminfo]
}
@ -715,8 +715,8 @@ namespace eval punk::cap::handlers::templates {
set itempath [dict get $iteminfo item]
set baseinfo [dict get $iteminfo baseinfo]
if {![dict exists $seen_dict $oname]} {
dict set seen_dict $oname 1
dict set itemdict $oname [list path $itempath {*}$baseinfo] ; #first seen of oname gets no number
dict set seen_dict $oname 1
dict set itemdict $oname [list path $itempath {*}$baseinfo] ; #first seen of oname gets no number
} else {
set n [dict get $seen_dict $oname]
incr n
@ -730,7 +730,7 @@ namespace eval punk::cap::handlers::templates {
set result [dict create]
set keys [lreverse [dict keys $itemdict]]
foreach k $keys {
set maybe ""
set maybe ""
foreach g $globsearches {
if {[string match $g $k]} {
set maybe $k
@ -745,7 +745,7 @@ namespace eval punk::cap::handlers::templates {
break
}
}
}
}
if {$maybe ne "" && $not eq ""} {
dict set result $k [dict get $itemdict $k]
}
@ -762,10 +762,10 @@ namespace eval punk::cap::handlers::templates {
# ++ +++ +++ +++ +++ +++ +++ +++ +++ +++ +++
## Ready
## Ready
package provide punk::cap::handlers::templates [namespace eval punk::cap::handlers::templates {
variable pkg punk::cap::handlers::templates
variable version
set version 999999.0a1.0
set version 999999.0a1.0
}]
return

2
src/modules/punk/char-999999.0a1.0.tm

@ -1181,7 +1181,7 @@ tcl::namespace::eval punk::char {
}
puts "ok.. loading"
set fd [open $file r]
fconfigure $fd -translation binary
chan configure $fd -translation binary
set data [read $fd]
close $fd
set block_count 0

2
src/modules/punk/config-0.1.tm

@ -32,7 +32,7 @@ tcl::namespace::eval punk::config {
if {$exename ne ""} {
set exefolder [file dirname $exename]
#default file logs to logs folder at same level as exe if writable, or empty string
set log_folder [file normalize $exefolder/../logs]
set log_folder [file normalize $exefolder/../logs] ;#~2ms
#tcl::dict::set startup scriptlib $exefolder/scriptlib
#tcl::dict::set startup apps $exefolder/../../punkapps

154
src/modules/punk/console-999999.0a1.0.tm

@ -783,7 +783,7 @@ namespace eval punk::console {
after cancel $timeoutid($callid)
set total_elapsed [expr {[clock millis] - $tslaunch($callid)}]
set last_elapsed [expr {[clock millis] - $lastvwait}]
set remaining [expr {$remaining - $last_elapsed}]
set remaining [expr {$remaining - $last_elapsed}]
if {$remaining < 0} {set remaining 0}
set newtime [expr {$remaining + $extension}]
set timeoutid($callid) [after $newtime [list set $waitvarname timedout]]
@ -797,7 +797,7 @@ namespace eval punk::console {
}
}
}
#response handler automatically removes it's own chan event
#response handler automatically removes it's own chan event
chan event $input readable {} ;#explicit remove anyway - review
if {$waitvar($callid) ne "timedout"} {
@ -814,7 +814,7 @@ namespace eval punk::console {
#it *might* be ok to restore entire state on an input channel
#(it's not always on all channels - e.g stdout has -winsize which is read-only)
#Safest to only restore what we think we've modified.
fconfigure $input -blocking [dict get $previous_input_state -blocking]
chan configure $input -blocking [dict get $previous_input_state -blocking]
@ -828,10 +828,10 @@ namespace eval punk::console {
set prefixdata [string range $input_read {*}$prefix_indices]
if {!$ignoreok && $prefixdata ne ""} {
#puts stderr "Warning - get_ansi_response_payload read extra data at start - '[ansistring VIEW -lf 1 $prefixdata]' (responsedata=[ansistring VIEW -lf 1 $responsedata])"
lappend input_chunks_waiting($input) $prefixdata
lappend input_chunks_waiting($input) $prefixdata
}
} else {
#timedout - or eof?
} else {
#timedout - or eof?
if {!$ignoreok} {
puts stderr "get_ansi_response_payload callid:$callid regex match '$capturingendregex' to input_read '[ansistring VIEW -lf 1 -vt 1 $input_read]' not found"
lappend input_chunks_waiting($input) $input_read
@ -872,11 +872,11 @@ namespace eval punk::console {
flush stdout
#concat and supply to existing handler in single text block - review
#Note will only
#Note will only
set waitingdata [join $input_chunks_waiting($input) ""]
set input_chunks_waiting($input) [list]
#after idle [list after 0 [list {*}$existing_handler $waitingdata]]
after idle [list {*}$existing_handler $waitingdata] ;#after 0 may be put ahead of events it shouldn't be - review
after idle [list {*}$existing_handler $waitingdata] ;#after 0 may be put ahead of events it shouldn't be - review
unset waitingdata
} else {
#! todo? for now, emit a clue as to what's happening.
@ -942,7 +942,7 @@ namespace eval punk::console {
#review - reading 1 byte at a time and repeatedly running the small capturing/completion regex seems a little inefficient... but we don't have a way to peek or put back chars (?)
#review (we do have the punk::console::input_chunks_waiting($chan) array to cooperatively put back data - but this won't work for user scripts not aware of this)
#review - timeout - what if terminal doesn't put data on stdin? error vs stderr report vs empty results
#review - Main loop may need to detect some terminal responses and store them for lookup instead-of or as-well-as this handler?
#review - Main loop may need to detect some terminal responses and store them for lookup instead-of or as-well-as this handler?
#e.g what happens to mouse-events while user code is executing?
#we may still need this handler if such a loop doesn't exist.
proc ansi_response_handler_regex {chan callid endregex} {
@ -973,14 +973,14 @@ namespace eval punk::console {
chan event $chan readable {}
set waits($callid) ok
} else {
# 30ms 16ms?
# 30ms 16ms?
set tsnow [clock millis]
set total_elapsed [expr {[set tslaunch($callid)] - $tsnow}]
set last_elapsed [expr {[set tsclock($callid)] - $tsnow}]
if {[string length $chunks($callid)] % 10 == 0 || $last_elapsed > 16} {
if {$total_elapsed > 3000} {
#REVIEW
#too long since initial read handler launched..
#too long since initial read handler launched..
#is other data being pumped into stdin? Eventloop starvation? Did we miss our codes?
#For now we'll stop extending the timeout.
after cancel $::punk::console::ansi_response_timeoutid($callid)
@ -1009,7 +1009,7 @@ namespace eval punk::console {
chan event $chan readable {}
# Something else
puts stderr "ansi_response_handler_regex Situation shouldn't be possible. No error and no bytes read on channel $chan but not chan blocked or EOF"
set waits($callid) error_unknown_zerobytes_while_not_blocked_or_eof
set waits($callid) error_unknown_zerobytes_while_not_blocked_or_eof
}
}
} ;#end namespace eval internal
@ -1034,7 +1034,7 @@ namespace eval punk::console {
if {$ansi_wanted <= 0} {
return
}
#a and a+ are called a *lot* - avoid even slight overhead of tailcall as it doesn't give us anything useful here
#a and a+ are called a *lot* - avoid even slight overhead of tailcall as it doesn't give us anything useful here
#tailcall punk::ansi::a+ {*}$args
::punk::ansi::a+ {*}$args
}
@ -1092,7 +1092,7 @@ namespace eval punk::console {
}
default {
set ansi_wanted 2
}
}
default {
error "punk::console::ansi expected 0|1|on|off|true|false|yes|no|default"
}
@ -1133,9 +1133,9 @@ namespace eval punk::console {
}
#test - find a better place to set terminal type
variable is_vt52 0
variable is_vt52 0
proc vt52 {{onoff {}}} {
#todo - return to colour state beforehand?. support 0-15 vt52 colours?
#todo - return to colour state beforehand?. support 0-15 vt52 colours?
#we shouldn't have to trun off colour to enter vt52 - we should make punk::console emit correct codes
variable is_vt52
if {$onoff eq ""} {
@ -1146,7 +1146,7 @@ namespace eval punk::console {
}
if {$is_vt52} {
if {!$onoff} {
puts -nonewline "\x1b<"
puts -nonewline "\x1b<"
set is_vt52 0
colour on
}
@ -1156,7 +1156,7 @@ namespace eval punk::console {
set is_vt52 1
colour off
} else {
puts -nonewline "\x1b<"
puts -nonewline "\x1b<"
#emit even though our is_vt52 flag thinks it's on. Should be harmless if underlying terminal already vt100+
}
}
@ -1222,10 +1222,10 @@ namespace eval punk::console {
return $onoff
} else {
if {$onoff} {
{*}[auto_execok stty] echo
{*}[auto_execok stty] echo
return 1
} else {
{*}[auto_execok stty] -echo
{*}[auto_execok stty] -echo
return 0
}
}
@ -1259,7 +1259,7 @@ namespace eval punk::console {
set expected [dict get $opts -expected_ms]
set capturingregex {(((.*)))$} ;#capture entire response same as response-payload
set ts_start [clock millis]
set ts_start [clock millis]
set response [punk::console::internal::get_ansi_response_payload -ignoreok 1 -return dict -expected_ms $expected -terminal $inoutchannels $request $capturingregex]
set ts_end [clock millis]
puts stderr $response
@ -1273,7 +1273,7 @@ namespace eval punk::console {
# -- --- --- --- --- --- ---
#get_ansi_response functions
#review - can these functions sensibly be used on channels not attached to the local console?
#review - can these functions sensibly be used on channels not attached to the local console?
#ie can we default to {stdin stdout} but allow other channel pairs?
# -- --- --- --- --- --- ---
proc get_cursor_pos {{inoutchannels {stdin stdout}}} {
@ -1284,13 +1284,13 @@ namespace eval punk::console {
#e.g \033\[46;1R
set capturingregex {(.*)(\x1b\[([0-9]+;[0-9]+)R)$} ;#must capture prefix,entire-response,response-payload
set request "\033\[6n"
set request "\033\[6n"
set payload [punk::console::internal::get_ansi_response_payload -terminal $inoutchannels $request $capturingregex]
#some terminals fail to respond properly to \x1b\[6n but do respond to \x1b\[?6n and vice-versa :/
#todo - what?
#todo - what?
#often terminals that fail will just put the raw request code on stdin - we could detect that and then
#try the other?
return $payload
}
proc get_checksum_rect {id page t l b r {inoutchannels {stdin stdout}}} {
@ -1333,7 +1333,7 @@ namespace eval punk::console {
proc get_device_attributes {{inoutchannels {stdin stdout}}} {
#DA1
variable last_da1_result
#first element in result is the terminal's architectural class 61,62,63,64.. ?
#first element in result is the terminal's architectural class 61,62,63,64.. ?
#for vt100 we get things like: "ESC\[?1;0c"
#for vt102 "ESC\[?6c"
@ -1368,7 +1368,7 @@ namespace eval punk::console {
proc get_tabstops {{inoutchannels {stdin stdout}}} {
#DECTABSR \x1b\[2\$w
#response example " ^[P2$u9/17/25/33/41/49/57/65/73/81^[\ " (where ^[ is \x1b)
#set capturingregex {(.*)(\x1b\[P2$u()\x1b\[\\)}
#set capturingregex {(.*)(\x1b\[P2$u()\x1b\[\\)}
#set capturingregex {(.*)(\x1bP2$u((?:[0-9]+)*(?:\/[0-9]+)*)\x1b\\)$}
set capturingregex {(.*)(\x1bP2\$u(.*)\x1b\\)$}
set request "\x1b\[2\$w"
@ -1387,7 +1387,7 @@ namespace eval punk::console {
#either terminal failed to report - or none set.
set testw [test_char_width \t]
if {[string is integer -strict $testw]} {
return $testw
return $testw
}
#We don't support none - default to 8
return 8
@ -1397,7 +1397,7 @@ namespace eval punk::console {
if {[llength $tslist] == 1} {
set testw [test_char_width \t]
if {[string is integer -strict $testw]} {
return $testw
return $testw
}
return 8
} else {
@ -1441,7 +1441,7 @@ namespace eval punk::console {
set cell_size ""
set cell_size_fallback 10x20
#todo - change -inoutchannels to -terminalobject with prebuilt default
#todo - change -inoutchannels to -terminalobject with prebuilt default
punk::args::define {
@id -id ::punk::console::cell_size
@ -1450,7 +1450,7 @@ namespace eval punk::console {
newsize -default "" -help\
"character cell pixel dimensions WxH
or omit to query cell size."
}
}
proc cell_size {args} {
set argd [punk::args::get_by_id ::punk::console::cell_size $args]
set inoutchannels [dict get $argd opts -inoutchannels]
@ -1462,11 +1462,11 @@ namespace eval punk::console {
if {$cell_size eq ""} {
#not set - try to query terminal's overall dimensions
set pixeldict [punk::console::get_xterm_pixels $inoutchannels]
lassign $pixeldict _w sw _h sh
lassign $pixeldict _w sw _h sh
if {[string is integer -strict $sw] && [string is integer -strict $sh]} {
lassign [punk::console::get_size] _cols columns _rows rows
#review - is returned size in pixels always a multiple of rows and cols?
set w [expr {$sw / $columns}]
set w [expr {$sw / $columns}]
set h [expr {$sh / $rows}]
set cell_size ${w}x${h}
return $cell_size
@ -1511,7 +1511,7 @@ namespace eval punk::console {
return [expr {$payload in {Z K M}}]
}
#todo - determine cursor on/off state before the call to restore properly.
#todo - determine cursor on/off state before the call to restore properly.
proc get_size {{inoutchannels {stdin stdout}}} {
lassign $inoutchannels in out
#we can't reliably use [chan names] for stdin,stdout. There could be stacked channels and they may have a names such as file22fb27fe810
@ -1521,7 +1521,7 @@ namespace eval punk::console {
} else {
if {$is_eof} {
error "punk::console::get_size eof on output channel $out ([info level 1])"
}
}
}
#we don't need to care about the input channel if chan configure on the output can give us the info.
#short circuit ansi cursor movement method if chan configure supports the -winsize value
@ -1529,7 +1529,7 @@ namespace eval punk::console {
if {[dict exists $outconf -winsize]} {
#this mechanism is much faster than ansi cursor movements
#REVIEW check if any x-platform anomalies with this method?
#can -winsize key exist but contain erroneous info? We will check that we get 2 ints at least
#can -winsize key exist but contain erroneous info? We will check that we get 2 ints at least
lassign [dict get $outconf -winsize] cols lines
if {[string is integer -strict $cols] && [string is integer -strict $lines]} {
return [list columns $cols rows $lines]
@ -1542,7 +1542,7 @@ namespace eval punk::console {
} else {
if {$is_eof} {
error "punk::console::get_size eof on input channel $in ([info level 1])"
}
}
}
#keep out of catch - no point in even trying a restore move if we can't get start position - just fail here.
@ -1565,7 +1565,7 @@ namespace eval punk::console {
puts -nonewline $out [$func_coff][$movefunc 2000 2000]
lassign [get_cursor_pos_list $inoutchannels] lines cols
puts -nonewline $out [$movefunc $start_row $start_col][$func_con];flush stdout
set result [list columns $cols rows $lines]
set result [list columns $cols rows $lines]
} errM]} {
puts -nonewline $out [$movefunc $start_row $start_col]
puts -nonewline $out [$func_con]
@ -1578,7 +1578,7 @@ namespace eval punk::console {
#faster than get_size when it is using ansi mechanism - but uses cursor_save - which we may want to avoid if calling during another operation which uses cursor save/restore
proc get_size_cursorrestore {{inoutchannels {stdin stdout}}} {
lassign $inoutchannels in out
#we use the same shortcircuit mechanism as get_size to avoid ansi at all if the output channel will give us the info directly
#we use the same shortcircuit mechanism as get_size to avoid ansi at all if the output channel will give us the info directly
set outconf [chan configure $out]
if {[dict exists $outconf -winsize]} {
lassign [dict get $outconf -winsize] cols lines
@ -1592,8 +1592,8 @@ namespace eval punk::console {
#This issue also occurs when switching back from the alternate screen buffer - so perhaps that needs to be addressed elsewhere.
puts -nonewline $out [punk::ansi::cursor_off][punk::ansi::cursor_save_dec][punk::ansi::move 2000 2000]
lassign [get_cursor_pos_list $inoutchannels] lines cols
puts -nonewline $out [punk::ansi::cursor_restore][punk::console::cursor_on];flush $out
set result [list columns $cols rows $lines]
puts -nonewline $out [punk::ansi::cursor_restore][punk::console::cursor_on];flush $out
set result [list columns $cols rows $lines]
} errM]} {
puts -nonewline $out [punk::ansi::cursor_restore_dec]
puts -nonewline $out [punk::ansi::cursor_on]
@ -1611,14 +1611,14 @@ namespace eval punk::console {
set capturingregex {(.*)(\x1b\[8;([0-9]+;[0-9]+)t)$} ;#must capture prefix,entire-response,response-payload
set request "\x1b\[18t"
set payload [punk::console::internal::get_ansi_response_payload -terminal $inoutchannels $request $capturingregex]
lassign [split $payload {;}] rows cols
lassign [split $payload {;}] rows cols
return [list columns $cols rows $rows]
}
proc get_xterm_pixels {{inoutchannels {stdin stdout}}} {
set capturingregex {(.*)(\x1b\[4;([0-9]+;[0-9]+)t)$} ;#must capture prefix,entire-response,response-payload
set request "\x1b\[14t"
set payload [punk::console::internal::get_ansi_response_payload -terminal $inoutchannels $request $capturingregex]
lassign [split $payload {;}] height width
lassign [split $payload {;}] height width
return [list width $width height $height]
}
@ -1629,7 +1629,7 @@ namespace eval punk::console {
set payload [punk::console::internal::get_ansi_response_payload -terminal $inoutchannels $request $capturingregex]
return $payload
}
#Terminals generally default to LNM being reset (off) ie enter key sends a lone <cr>
#Terminals generally default to LNM being reset (off) ie enter key sends a lone <cr>
#Terminals tested on windows either don't respond to this query, or respond with 0 (meaning mode not understood)
#I presume from this that almost nobody is using LNM 1 (which sends both <cr> and <lf>)
proc get_mode_LNM {{inoutchannels {stdin stdout}}} {
@ -1689,7 +1689,7 @@ namespace eval punk::console {
#terminals lie. This should be a reasonable (albeit relatively slow) test of actual width - but some terminals seem to miscalculate.
#todo - a visual interactive test/questionnaire to ask user if things are lining up or if the terminal is telling fibs about cursor position.
#todo - determine if these anomalies are independent of font
#punk::ansi should be able to glean widths from unicode data files - but this may be incomplete - todo - compare with what terminal actually does.
#punk::ansi should be able to glean widths from unicode data files - but this may be incomplete - todo - compare with what terminal actually does.
#review - vertical movements (e.g /n /v will cause emit 0 to be ineffective - todo - disallow?)
proc test_char_width {char_or_string {emit 0}} {
#return 1
@ -1797,7 +1797,7 @@ namespace eval punk::console {
#don't set ansi_avaliable here - we want to be able to change things, retest etc.
if {"windows" eq "$::tcl_platform(platform)"} {
if {[package provide twapi] ne ""} {
set h_out [twapi::get_console_handle stdout]
set h_out [twapi::get_console_handle stdout]
set existing_mode [twapi::GetConsoleMode $h_out]
if {[expr {$existing_mode & 4}]} {
#virtual terminal processing happens to be enabled - so it's supported
@ -1808,12 +1808,12 @@ namespace eval punk::console {
#try temporarily setting it - if we get an error - ansi not supported
if {[catch {
twapi::SetConsoleMode $h_out [expr {$existing_mode | 4}]
twapi::SetConsoleMode $h_out [expr {$existing_mode | 4}]
} errM]} {
return 0
}
#restore
twapi::SetConsoleMode $h_out [expr {$existing_mode & ~4}]
twapi::SetConsoleMode $h_out [expr {$existing_mode & ~4}]
return 1
} else {
#todo - try a cursorpos query and read stdin to see if we got a response?
@ -1837,26 +1837,26 @@ namespace eval punk::console {
set ansi_available [test_can_ansi]
return $ansi_available
}
return 1
return 1
}
variable grapheme_cluster_support [dict create] ;#default empty dict for unknown/untested
variable grapheme_cluster_support [dict create] ;#default empty dict for unknown/untested
#todo - flag to retest? (for consoles where grapheme cluster support can be disabled e.g via decmode 2027)
proc grapheme_cluster_support {} {
variable grapheme_cluster_support
if {[dict size $grapheme_cluster_support]} {
return $grapheme_cluster_support
return $grapheme_cluster_support
}
if {[info exists ::env(TERM_PROGRAM)]} {
#terminals known to support grapheme clusters, but unable to respond to decmode request 2027
#wezterm (on windows as at 2024-12 decmode 2027 doesn't work)
#REVIEW - what if terminal is remote wezterm? can/will this env variable
#REVIEW - what if terminal is remote wezterm? can/will this env variable
# iterm and apple terminal also set TERM_PROGRAM
if {[string tolower $::env(TERM_PROGRAM)] in [list wezterm]} {
set is_available 1
return [dict create available 1 mode set]
return [dict create available 1 mode set]
}
}
#where 1 = set, 2 = unset. (0 = mode not recognised, 3 = permanently set, 4 = permanently unset)
@ -1884,7 +1884,7 @@ namespace eval punk::console {
set m "BAD_RESPONSE"
}
}
return [dict create available $is_available mode $m]
return [dict create available $is_available mode $m]
}
@ -1947,7 +1947,7 @@ namespace eval punk::console {
set was_raw 1
}
puts -nonewline stdout \033\[6n ;flush stdout
fconfigure stdin -blocking 0
chan configure stdin -blocking 0
set info [read stdin 20] ;#
after 1
if {[string first "R" $info] <=0} {
@ -2015,8 +2015,8 @@ namespace eval punk::console {
(aka: cursor home)
The sequence emitted will depend on the mode of the
terminal as stored in the consolehandle.
Directly setting the mode via raw escape sequences:
terminal as stored in the consolehandle.
Directly setting the mode via raw escape sequences:
e.g unset_mode DECANM for vt52
or puts \x1b< to return to ANSI
will not necessarily update the application of
@ -2036,7 +2036,7 @@ namespace eval punk::console {
This sequence will generally not be understood by
terminals that are not in vt52 mode even if higher
modes are supported.
}
@values -min 2 -max 2
row -type integer -help\
@ -2045,7 +2045,7 @@ namespace eval punk::console {
"column number - starting at 1"
}]
proc move {row col} {
upvar ::punk::console::is_vt52 is_vt52
upvar ::punk::console::is_vt52 is_vt52
if {!$is_vt52} {
return [punk::ansi::move $row $col]
} else {
@ -2053,7 +2053,7 @@ namespace eval punk::console {
}
}
proc move_forward {n} {
upvar ::punk::console::is_vt52 is_vt52
upvar ::punk::console::is_vt52 is_vt52
if {!$is_vt52} {
puts -nonewline stdout [punk::ansi::move_forward $n]
} else {
@ -2061,7 +2061,7 @@ namespace eval punk::console {
}
}
proc move_back {n} {
upvar ::punk::console::is_vt52 is_vt52
upvar ::punk::console::is_vt52 is_vt52
if {!$is_vt52} {
puts -nonewline stdout [punk::ansi::move_back $n]
} else {
@ -2075,7 +2075,7 @@ namespace eval punk::console {
puts -nonewline stdout [punk::ansi::move_down $n]
}
proc move_column {col} {
upvar ::punk::console::is_vt52 is_vt52
upvar ::punk::console::is_vt52 is_vt52
if {!$is_vt52} {
puts -nonewline stdout [punk::ansi::move_column $col]
} else {
@ -2086,7 +2086,7 @@ namespace eval punk::console {
puts -nonewline stdout [punk::ansi::move_row $row]
}
proc move_emit {row col data args} {
upvar ::punk::console::is_v52 is_vt52
upvar ::punk::console::is_v52 is_vt52
if {!$is_vt52} {
puts -nonewline stdout [punk::ansi::move_emit $row $col $data {*}$args]
} else {
@ -2226,7 +2226,7 @@ namespace eval punk::console {
}
proc titleset {windowtitle} {
puts -nonewline stdout [punk::ansi::titleset $windowtitle]
}
}
proc test_decaln {} {
puts -nonewline stdout [punk::ansi::test_decaln]
}
@ -2239,10 +2239,10 @@ namespace eval punk::console {
if { $ansi_wanted <= 0} {
punk::console::local::titleset $windowtitle
} else {
ansi::titleset $windowtitle
ansi::titleset $windowtitle
}
}
#no known pure-ansi solution
#no known pure-ansi solution
proc titleget {} {
return [local::titleget]
}
@ -2272,14 +2272,14 @@ namespace eval punk::console {
#experimental
proc rhs_prompt {col text} {
package require textblock
lassign [textblock::size $text] _w tw _h th
lassign [textblock::size $text] _w tw _h th
if {$th > 1} {
#move up first.. need to know current line?
}
#set blanks [string repeat " " [expr {$col + $tw}]]
#puts -nonewline [punk::ansi::erase_eol]$blanks;move_emit_return this $col $text
#puts -nonewline [move_emit_return this $col [punk::ansi::insert_spaces 150]$text]
cursor_save_dec
cursor_save_dec
#move_emit_return this $col [punk::ansi::move_forward 50][punk::ansi::insert_spaces 150][punk::ansi::move_back 50][punk::ansi::move_forward $col]$text
#puts -nonewline [punk::ansi::insert_spaces 150][punk::ansi::move_column $col]$text
puts -nonewline [punk::ansi::erase_eol][punk::ansi::move_column $col]$text
@ -2323,7 +2323,7 @@ namespace eval punk::console {
18 30 60 C0 60 30 18 00
00 00 7E 00 7E 00 00 00
60 30 18 0C 18 30 60 00
3C 66 0C 18 18 00 18 00
3C 66 0C 18 18 00 18 00
}
#libungif extras
append fontmap1 {
@ -2491,7 +2491,7 @@ namespace eval punk::console {
#curses attr off reverse
#a noreverse
set reverse 0
set output ""
set output ""
set charno 0
foreach char [split $str {}] {
binary scan $char c f
@ -2528,9 +2528,9 @@ namespace eval punk::console {
}
proc display {} {
lassign [punk::console::get_cursor_pos_list] orig_row orig_col
punk::console::move 20 20
punk::console::move 20 20
punk::console::clear_above
punk::console::move 0 0
punk::console::move 0 0
puts -nonewline [bigstr [clock format [clock seconds] -format %H:%M:%S] 10 5]
punk::console::move $orig_row $orig_col
@ -2539,9 +2539,9 @@ namespace eval punk::console {
proc displaystr {str} {
lassign [punk::console::get_cursor_pos_list] orig_row orig_col
punk::console::move 20 20
punk::console::move 20 20
punk::console::clear_above
punk::console::move 0 0
punk::console::move 0 0
puts -nonewline [bigstr $str 10 5]
punk::console::move $orig_row $orig_col
@ -2571,13 +2571,13 @@ namespace eval punk::console {
if {$dingbat_heavy_plus_width == 2} {
set can_terminal_report_dingbat_width 1
} else {
puts stderr "punk::console warning: terminal either not displaying wide unicode as wide, or unable to report width properly."
puts stderr "punk::console warning: terminal either not displaying wide unicode as wide, or unable to report width properly."
}
set diacritic_width [punk::console::test_char_width a\u0300]
if {$diacritic_width == 1} {
set can_terminal_report_diacritic_width 1
} else {
puts stderr "punk::console warning: terminal unable to report diacritic width properly."
puts stderr "punk::console warning: terminal unable to report diacritic width properly."
}
if {$can_high_unicode && $can_regex_high_unicode && $can_terminal_report_dingbat_width && $can_terminal_report_diacritic_width} {
@ -2617,7 +2617,7 @@ namespace eval punk::console::check {
}
return $has_bug_legacysymbolwidth
}
return 1
return 1
}
variable has_bug_zwsp -1 ;#undetermined
proc has_bug_zwsp {} {

203
src/modules/punk/fileline-999999.0a1.0.tm

@ -9,7 +9,7 @@
# @@ Meta Begin
# Application punk::fileline 999999.0a1.0
# Meta platform tcl
# Meta license BSD
# Meta license BSD
# @@ Meta End
@ -20,7 +20,7 @@
#[manpage_begin punkshell_module_punk::fileline 0 999999.0a1.0]
#[copyright "2024"]
#[titledesc {file line-handling utilities}] [comment {-- Name section and table of contents description --}]
#[moddesc {punk fileline}] [comment {-- Description at end of page heading --}]
#[moddesc {punk fileline}] [comment {-- Description at end of page heading --}]
#[require punk::fileline]
#[keywords module text parse file encoding BOM]
#[description]
@ -33,7 +33,7 @@
#[para]Utilities for in-memory analysis of text file data as both line data and byte/char-counted data whilst preserving the line-endings (even if mixed)
#[para]This is important for certain text files where examining the number of chars/bytes is important
#[para]For example - windows .cmd/.bat files need some byte counting to determine if labels lie on chunk boundaries and need to be moved.
#[para]This chunk-size counting will depend on the character encoding.
#[para]This chunk-size counting will depend on the character encoding.
#[para]Despite including the word 'file', the library doesn't necessarily deal with reading/writing to the filesystem -
#[para]The raw data can be supplied as a string, or loaded from a file using punk::fileline::get_textinfo -file <filename>
#[subsection Concepts]
@ -42,13 +42,13 @@
# package require punk::fileline
# package require fileutil
# set rawdata [lb]fileutil::cat data.txt -translation binary[rb]
# punk::fileline::class::textinfo create obj_data $rawdata
# punk::fileline::class::textinfo create obj_data $rawdata
# puts stdout [lb]obj_data linecount[rb]
#[example_end]
#[subsection Notes]
#[para]Line records are referred to by a zero-based index instead of a one-based index as is commonly used when displaying files.
#[para]This is for programming consistency and convenience, and the module user should do their own conversion to one-based indexing for line display or messaging if desired.
#[para]No support for lone carriage-returns being interpreted as line-endings.
#[para]No support for lone carriage-returns being interpreted as line-endings.
#[para]CR line-endings that are intended to be interpreted as such should be mapped to something else before the data is supplied to this module.
# ++ +++ +++ +++ +++ +++ +++ +++ +++ +++ +++
@ -141,7 +141,7 @@ namespace eval punk::fileline::class {
variable o_line_epoch
variable o_payloadlist
variable o_linemap
variable o_LF_C
variable o_LF_C
variable o_CRLF_C
@ -158,7 +158,7 @@ namespace eval punk::fileline::class {
#[para] Constructor for textinfo object which represents a chunk or all of a file
#[para] datachunk should be passed with the file data including line-endings as-is for full functionality. ie use something like:
#[example_begin]
# fconfigure $fd -translation binary
# chan configure $fd -translation binary
# set chunkdata [lb]read $fd[rb]]
#or
# set chunkdata [lb]fileutil::cat <filename> -translation binary[rb]
@ -191,7 +191,7 @@ namespace eval punk::fileline::class {
set o_bom "" ;#review
set o_chunk $datachunk
set o_line_epoch [list]
set o_line_epoch [list]
set o_chunk_epoch [list "fromchunkchange-at-[clock micros]"]
set crlf_lf_placeholders [list \uFFFF \uFFFE] ;#defaults - if already exist in file - error out with message
set defaults [dict create\
@ -206,11 +206,11 @@ namespace eval punk::fileline::class {
}
}
set opts [dict merge $defaults $args]
# -- --- --- --- --- --- ---
# -- --- --- --- --- --- ---
set opt_substitutionmap [dict get $opts -substitutionmap] ;#review - can be done by caller - or a loadable -policy
set opt_crlf_lf_placeholders [dict get $opts -crlf_lf_placeholders]
set opt_userid [dict get $opts -userid]
# -- --- --- --- --- --- ---
# -- --- --- --- --- --- ---
if {[llength $opt_crlf_lf_placeholders] != 2 || [string length [lindex $opt_crlf_lf_placeholders 0]] !=1 || [string length [lindex $opt_crlf_lf_placeholders 1]] !=1} {
error "textinfo::constructor error: -crlf_lf_placeholders requires a list of exactly 2 chars"
@ -261,7 +261,7 @@ namespace eval punk::fileline::class {
#[call class::textinfo [method chunk] [arg chunkstart] [arg chunkend]]
#[para]Return a range of bytes from the underlying raw chunk data.
#[para] e.g The following retrieves the entire chunk
#[para] objName chunk 0 end
#[para] objName chunk 0 end
return [string range $o_chunk $chunkstart $chunkend]
}
method chunklen {} {
@ -273,7 +273,7 @@ namespace eval punk::fileline::class {
method chunk_boundary_display {chunkstart chunkend chunksize args} {
#*** !doctools
#[call class::textinfo [method chunk_boundary_display]]
#[para]Returns a string displaying the boundaries at chunksize bytes between chunkstart and chunkend
#[para]Returns a string displaying the boundaries at chunksize bytes between chunkstart and chunkend
#[para]Defaults to using ansi colour if punk::ansi module is available. Use -ansi 0 to disable colour
set opts [dict create\
-ansi $::punk::fileline::ansi::enabled\
@ -331,7 +331,7 @@ namespace eval punk::fileline::class {
if {$opt_ansi} {
set ::punk::fileline::ansi::enabled 1
} else {
set ::punk::fileline::ansi::enabled 0
set ::punk::fileline::ansi::enabled 0
}
if {"::punk::fileline::ansistrip" ne [info commands ::punk::fileline::ansistrip]} {
proc ::punk::fileline::a {args} {
@ -350,7 +350,7 @@ namespace eval punk::fileline::class {
}
proc ::punk::fileline::ansistrip {str} {
if {$::punk::fileline::ansi::enabled} {
tailcall ::punk::fileline::ansi::ansistrip $str
tailcall ::punk::fileline::ansi::ansistrip $str
} else {
return $str
}
@ -361,10 +361,10 @@ namespace eval punk::fileline::class {
#suport simple end+-int (+-)start(+-)int to set linebase to line corresponding to chunkstart or chunkend
#also simple int+int and int-int - nothing more complicated (similar to Tcl lrange etc in that regard)
#commonly this will be something like -start or -end
#commonly this will be something like -start or -end
if {![string is integer -strict $opt_linebase]} {
set sign ""
set errunrecognised "unrecognised -linebase value '$opt_linebase'. Expected positive or negative integer or -start -start-int -start+int -end -end-int -end+int or -eof (where leading - is optional but probably desirable) "
set errunrecognised "unrecognised -linebase value '$opt_linebase'. Expected positive or negative integer or -start -start-int -start+int -end -end-int -end+int or -eof (where leading - is optional but probably desirable) "
if {[string index $opt_linebase 0] eq "-"} {
set sign -
set tail [string range $opt_linebase 1 end]
@ -402,7 +402,7 @@ namespace eval punk::fileline::class {
} else {
set linebase $maxline
}
set linebase ${sign}$linebase
set linebase ${sign}$linebase
} elseif {[string match start* $tail]} {
set endmath [string range $tail 5 end]
if {[string length $endmath]} {
@ -489,7 +489,7 @@ namespace eval punk::fileline::class {
set j [expr {$i+1}]
append result [string map [list %b% $b %i% $i %j% $j] $opt_boundaryheader] \n
}
set low [expr {max(($b - $pre_bytes),0)}]
set low [expr {max(($b - $pre_bytes),0)}]
set high [expr {min(($b + $post_bytes),$max_bytes)}]
set lineinfolist [my chunkrange_to_lineinfolist $low $high -show_truncated 1]
@ -503,11 +503,11 @@ namespace eval punk::fileline::class {
set e [dict get $lineinfo end]
set boundarymarker ""
set displayidx ""
set displayidx ""
set linenum_display $linenum
if {$s <= $b && $e >= $b} {
set idx [expr {$b - $s}] ;#index into whole position in whole line - not so useful if we're viewing a small section of a line
set char [string index [my line $lineidx] $idx]
set char [string index [my line $lineidx] $idx]
set char_display [string map [list \r <CR> \n <LF>] $char]
if {[dict get $lineinfo is_truncated]} {
set tside [dict get $lineinfo truncatedside]
@ -527,29 +527,29 @@ namespace eval punk::fileline::class {
set linenum_display ${linenum_display},$idx
}
set lhs_status $opt_cmark ;#default
set lhs_status $opt_cmark ;#default
set rhs_status $opt_cmark ;#default
if {[dict get $lineinfo is_truncated]} {
set line [dict get $lineinfo truncated]
set tside [dict get $lineinfo truncatedside]
if {"left" in $tside && "right" in $tside } {
set lhs_status $opt_tmark
set rhs_status $opt_tmark
set lhs_status $opt_tmark
set rhs_status $opt_tmark
} elseif {"left" in $tside} {
set lhs_status $opt_tmark
set lhs_status $opt_tmark
} elseif {"right" in $tside} {
set rhs_status $opt_tmark
}
} else {
set line [my line $lineidx]
set line [my line $lineidx]
}
if {$displayidx ne ""} {
set line [string replace $line $displayidx $displayidx [a+ White green bold]$char_display[a]]
set line [string replace $line $displayidx $displayidx [a+ White green bold]$char_display[a]]
}
set displayline [string map $le_map $line]
lappend result_list [list $linenum_display $boundarymarker $lhs_status $displayline $rhs_status]
set displayline [string map $le_map $line]
lappend result_list [list $linenum_display $boundarymarker $lhs_status $displayline $rhs_status]
}
set title_linenum "LNUM"
set linenums [lsearch -index 0 -all -inline -subindices $result_list *]
@ -586,12 +586,12 @@ namespace eval punk::fileline::class {
method line {lineindex} {
#*** !doctools
#[call class::textinfo [method line] [arg lineindex]]
#[para]Reconstructs and returns the raw line using the payload and per-line stored line-ending metadata
#[para]Reconstructs and returns the raw line using the payload and per-line stored line-ending metadata
#[para]A 'line' may be returned without a line-ending if the unerlying chunk had trailing data without a line-ending (or the chunk was loaded under a non-standard -policy setting)
#[para]Whilst such data may not conform to definitions (e.g POSIX) of the terms 'textfile' and 'line' - it is useful here to represent it as a line with metadata le set to "none"
#[para]To return just the data which might more commonly be needed for dealing with lines, use the [method linepayload] method - which returns the line data minus line-ending
lassign [my numeric_linerange $lineindex 0] lineindex
lassign [my numeric_linerange $lineindex 0] lineindex
set le [dict get $o_linemap $lineindex le]
set le_chars [dict get [dict create lf \n crlf \r\n none ""] $le]
@ -641,13 +641,13 @@ namespace eval punk::fileline::class {
set opt_strategy [dict get $opts -strategy]
# -- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- ---
set opt_start [dict get $opts -start]
set opt_start [expr {$opt_start}]
set opt_start [expr {$opt_start}]
if {$opt_start != 0} {error "-start unimplemented"}
# -- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- ---
set opt_end [dict get $opts -end]
set max_line_index [expr {[llength $o_payloadlist]-1}]
if {$opt_end eq "end"} {
set opt_end $max_line_index
set opt_end $max_line_index
}
#TODO
if {$opt_end < $max_line_index} {error "-end less than max_line_index unimplemented"}
@ -705,7 +705,7 @@ namespace eval punk::fileline::class {
#[para]Line Metadata such as the line-ending for a particular line and the byte/character range it occupies within the chunk can be retrieved with the [method linemeta] method
#[para]To retrieve both the line text and metadata in a single call the [method lineinfo] method can be used
#[para]To retrieve an entire line including line-ending use the [method line] method.
lassign [my numeric_linerange $lineindex 0] lineindex
lassign [my numeric_linerange $lineindex 0] lineindex
return [lindex $o_payloadlist $lineindex]
}
method linepayloads {startindex endindex} {
@ -722,17 +722,17 @@ namespace eval punk::fileline::class {
#[list_begin itemized]
#[item] le
#[para] A string representing the type of line-ending: crlf|lf|none
#[item] linelen
#[item] linelen
#[para] The number of characters/bytes in the whole line including line-ending if any
#[item] payloadlen
#[item] payloadlen
#[para] The number of character/bytes in the line excluding line-ending
#[item] start
#[para] The zero-based index into the associated raw file data indicating at which byte/character index this line begins
#[para] The zero-based index into the associated raw file data indicating at which byte/character index this line begins
#[item] end
#[para] The zero-based index into the associated raw file data indicating at which byte/character index this line ends
#[para] This end-point corresponds to the last character of the line-ending if any - not necessarily the last character of the line's payload
#[list_end]
lassign [my numeric_linerange $lineindex 0] lineindex
lassign [my numeric_linerange $lineindex 0] lineindex
dict get $o_linemap $lineindex
}
method lineinfo {lineindex} {
@ -797,7 +797,7 @@ namespace eval punk::fileline::class {
method chunkrange_to_linerange {chunkstart chunkend} {
#*** !doctools
#[call class::textinfo [method chunkrange_to_linerange] [arg chunkstart] [arg chunkend]]
lassign [my numeric_chunkrange $chunkstart $chunkend] chunkstart chunkend
lassign [my numeric_chunkrange $chunkstart $chunkend] chunkstart chunkend
set linestart -1
for {set i 0} {$i < [llength $o_payloadlist]} {incr i} {
@ -829,7 +829,7 @@ namespace eval punk::fileline::class {
#[para]truncation shows the shortened (missing bytes on left and/or right side) part of the entire line (potentially including line-ending or even partial line-ending)
#[para]Note that this truncation info is only in the return value of this method - and will not be reflected in [method lineinfo] queries to the main chunk.
lassign [my numeric_chunkrange $chunkstart $chunkend] chunkstart chunkend
lassign [my numeric_chunkrange $chunkstart $chunkend] chunkstart chunkend
set defaults [dict create\
-show_truncated 0\
]
@ -840,9 +840,9 @@ namespace eval punk::fileline::class {
}
}
set opts [dict merge $defaults $args]
# -- --- --- --- --- --- --- ---
# -- --- --- --- --- --- --- ---
set opt_show_truncated [dict get $opts -show_truncated]
# -- --- --- --- --- --- --- ---
# -- --- --- --- --- --- --- ---
set infolist [list]
set linerange [my chunkrange_to_linerange $chunkstart $chunkend]
@ -878,8 +878,8 @@ namespace eval punk::fileline::class {
set truncated [string range $payload_and_le $split end]
set lhs [string range $payload_and_le 0 $split-1]
dict set first truncated $truncated
dict set first truncatedleft $lhs
dict set first truncated $truncated
dict set first truncatedleft $lhs
}
}
###########################
@ -908,7 +908,7 @@ namespace eval punk::fileline::class {
if {$chunkend < [dict get $end_info end]} {
#there is rhs truncation
if {[dict get $first is_truncated]} {
dict set first truncatedside [list left right]
dict set first truncatedside [list left right]
} else {
dict set first is_truncated 1
dict set first truncatedside [list right]
@ -925,7 +925,7 @@ namespace eval punk::fileline::class {
set le_chars [dict get [dict create lf \n crlf \r\n none ""] [dict get $end_info le]]
set payload_and_le "${payload}${le_chars}"
set split [expr {$chunkend - $line_start}]
set truncated [string range $payload_and_le 0 $split]
set truncated [string range $payload_and_le 0 $split]
set rhs [string range $payload_and_le $split+1 end]
dict set first truncatedright $rhs
if {"left" ni [dict get $first truncatedside]} {
@ -971,13 +971,13 @@ namespace eval punk::fileline::class {
set payload_and_le "${payload}${le_chars}"
set split [expr {$chunkend - $line_start}]
set truncated [string range $payload_and_le 0 $split]
set truncated [string range $payload_and_le 0 $split]
set rhs [string range $payload_and_le $split+1 end]
dict set last truncated $truncated
dict set last truncatedright $rhs
#this has the effect that truncating the rhs by 1 can result in truncated being larger than original payload for crlf lines - as payload now sees the cr
#this is a bit unintuitive - but probably best reflects the reality. The truncated value is the truncated 'line' rather than the truncated 'payload'
#this is a bit unintuitive - but probably best reflects the reality. The truncated value is the truncated 'line' rather than the truncated 'payload'
}
}
@ -991,7 +991,7 @@ namespace eval punk::fileline::class {
###########################
#assertion all records have is_truncated key.
#assertion if is_truncated == 1 truncatedside should contain a list of either left, right or both left and right
#assertion If not opt_show_truncated - then truncated records will not have truncated,truncatedleft,truncatedright keys.
#assertion If not opt_show_truncated - then truncated records will not have truncated,truncatedleft,truncatedright keys.
return $infolist
}
@ -1017,12 +1017,12 @@ namespace eval punk::fileline::class {
#Also check if the truncation is directly between an crlf
#both an lhs split and an rhs split could land between cr and lf
#to be precise - we should presumably count the part within our chunk as either a none for cr or an lf
#This means a caller counting chunk by chunk using this method will sometimes get the wrong answer depending on where crlfs lie relative to their chosen chunk size
#This means a caller counting chunk by chunk using this method will sometimes get the wrong answer depending on where crlfs lie relative to their chosen chunk size
#This is presumably ok - as it should be a well known thing to watch out for.
#If we're only receiving chunk by chunk we can't reliably detect splits vs lone <cr>s in the data
#There are surely more efficient ways for a caller to count line-endings in the way that makes sense for them
#but we should makes things as easy as possible for users of this line/chunk structure anyway.
set first [lindex $infolines 0]
if {[dict get $first is_truncated]} {
#could be the only line - and truncated at one or both ends.
@ -1035,7 +1035,7 @@ namespace eval punk::fileline::class {
#if so - then split can only be left side
}
return [dict create lf $lf_count crlf $crlf_count unterminated $none_count warning line_ending_splits_unimplemented]
}
@ -1061,13 +1061,13 @@ namespace eval punk::fileline::class {
method normalize_indices {startidx endidx max} {
#*** !doctools
#[call class::textinfo [method normalize_indices] [arg startidx] [arg endidx] [arg max]]
#[para]A utility to convert some of the of Tcl-style list-index expressions such as end, end-1 etc to valid indices in the range 0 to the supplied max
#[para]Basic addition and subtraction expressions such as 4-1 5+2 are accepted
#[para]A utility to convert some of the of Tcl-style list-index expressions such as end, end-1 etc to valid indices in the range 0 to the supplied max
#[para]Basic addition and subtraction expressions such as 4-1 5+2 are accepted
#[para]startidx higher than endidx is allowed
#[para]Unlike Tcl's index expressions - we raise an error if the calculated index is out of bounds 0 to max
#[para]Unlike Tcl's index expressions - we raise an error if the calculated index is out of bounds 0 to max
set original_startidx $startidx
set original_endidx $endidx
set startidx [string map [list _ ""] $startidx] ;#don't barf on Tcl 8.7+ underscores in numbers - we can't just use expr because it will not handle end-x
set startidx [string map [list _ ""] $startidx] ;#don't barf on Tcl 8.7+ underscores in numbers - we can't just use expr because it will not handle end-x
set endidx [string map [list _ ""] $endidx]
if {![string is digit -strict "$startidx$endidx"]} {
foreach whichvar [list start end] {
@ -1078,9 +1078,9 @@ namespace eval punk::fileline::class {
set index $max
}
"*-*" {
#end-int or int-int - like lrange etc we don't accept arbitrarily complex expressions
#end-int or int-int - like lrange etc we don't accept arbitrarily complex expressions
lassign [split $index -] A B
if {$A eq "end"} {
if {$A eq "end"} {
set index [expr {$max - $B}]
} else {
set index [expr {$A - $B}]
@ -1088,7 +1088,7 @@ namespace eval punk::fileline::class {
}
"*+*" {
lassign [split $index +] A B
if {$A eq "end"} {
if {$A eq "end"} {
#review - this will just result in out of bounds error in final test - as desired
#By calculating here - we will see the result in the error message - but it's probably not particularly useful - as we don't really need end+ support at all.
set index [expr {$max + $B}]
@ -1098,9 +1098,9 @@ namespace eval punk::fileline::class {
}
default {
#May be something like +2 or -0 which braced expr can hanle
#we would like to avoid unbraced expr here - as we're potentially dealing with ranges that may come from external sources.
#we would like to avoid unbraced expr here - as we're potentially dealing with ranges that may come from external sources.
if {[catch {expr {$index}} index]} {
#could be end+x - but we don't want out of bounds to be valid
#could be end+x - but we don't want out of bounds to be valid
#set it to something that the final bounds expr test can deal with
set index Inf
}
@ -1109,13 +1109,13 @@ namespace eval punk::fileline::class {
}
}
}
#Unlike Tcl lrange,lindex etc - we don't want to support out of bound indices.
#Unlike Tcl lrange,lindex etc - we don't want to support out of bound indices.
#show the supplied index and what it was mapped to in the error message.
if {$startidx < 0 || $startidx > $max} {
error "Bad start index '$original_startidx'. $startidx out of bounds 0 - $max"
error "Bad start index '$original_startidx'. $startidx out of bounds 0 - $max"
}
if {$endidx < 0 || $endidx > $max} {
error "Bad end index '$original_endidx'. $endidx out of bounds 0 - $max (try $max or end)"
error "Bad end index '$original_endidx'. $endidx out of bounds 0 - $max (try $max or end)"
}
return [list $startidx $endidx]
}
@ -1136,7 +1136,7 @@ namespace eval punk::fileline::class {
set crlf_replace [list \r\n $o_CRLF_C \n $o_LF_C]
set normalised_data [string map $crlf_replace $o_chunk]
set lf_lines [split $normalised_data $o_LF_C]
set lf_lines [split $normalised_data $o_LF_C]
set idx 0
set lf_count 0
@ -1145,14 +1145,14 @@ namespace eval punk::fileline::class {
set i 0
set imax [expr {[llength $lf_lines]-1}]
foreach lfln $lf_lines {
set crlf_parts [split $lfln $o_CRLF_C]
set crlf_parts [split $lfln $o_CRLF_C]
if {[llength $crlf_parts] <= 1} {
#no crlf
set payloadlen [string length $lfln]
set le_size 1
set le lf
if {$i == $imax} {
#no more lf segments - and no crlfs
#no more lf segments - and no crlfs
if {$payloadlen > 0} {
#last line in split has chars - therefore there was no trailing line-ending
set le_size 0
@ -1177,7 +1177,7 @@ namespace eval punk::fileline::class {
set payloadlen [string length $crlfpart]
set linelen [expr {$payloadlen + 2}]
dict set o_linemap $idx [list le crlf linelen $linelen payloadlen $payloadlen start $filedata_offset end [expr {$filedata_offset + $linelen -1}]]
incr filedata_offset $linelen
incr filedata_offset $linelen
incr crlf_count
incr idx
}
@ -1200,7 +1200,7 @@ namespace eval punk::fileline::class {
set le lf
}
lappend o_payloadlist $lfpart
lappend o_payloadlist $lfpart
set linelen [expr {$payloadlen + $le_size}]
dict set o_linemap $idx [list le $le linelen $linelen payloadlen $payloadlen start $filedata_offset end [expr {$filedata_offset + $linelen -1}]]
incr filedata_offset $linelen
@ -1221,8 +1221,11 @@ namespace eval punk::fileline::class {
#o_linemap
set oldsize [string length $o_chunk]
set newchunk ""
#review - what was the intention here?
puts stderr "regenerate_chunk -warning code incomplete"
dict for {idx lineinfo} $o_linemap {
set
#???
#set
}
@ -1248,19 +1251,19 @@ namespace eval punk::fileline {
#*** !doctools
#[subsection {Namespace punk::fileline}]
#[para] Core API functions for punk::fileline
#[para] Core API functions for punk::fileline
#[list_begin definitions]
punk::args::define {
punk::args::define {
@id -id ::punk::fileline::get_textinfo
@cmd -name punk::fileline::get_textinfo -help\
"return: textinfo object instance"
-file -default {} -type existingfile
-translation -default iso8859-1
-translation -default iso8859-1
-encoding -default "\uFFFF"
-includebom -default 0
@values -min 0 -max 1
}
}
proc get_textinfo {args} {
#*** !doctools
#[call get_textinfo [opt {option value...}] [opt datachunk]]
@ -1272,7 +1275,7 @@ namespace eval punk::fileline {
#[para]If -includebom 1 is specified - the bom will be retained in the stored chunk and the data for line 1, but will undergo the same encoding transformation as the rest of the data
#[para]The get_bomid method of the returned object will contain an identifier for any BOM encountered.
#[para] e.g utf-8,utf-16be, utf-16le, utf-32be, utf32-le, SCSU, BOCU-1,GB18030, UTF-EBCDIC, utf-1, utf-7
#[para]If the encoding specified in the BOM isn't recognised by Tcl - the resulting data is likely to remain as the raw bytes of whatever encoding that is.
#[para]If the encoding specified in the BOM isn't recognised by Tcl - the resulting data is likely to remain as the raw bytes of whatever encoding that is.
#[para]Currently only utf-8, utf-16* and utf-32* are properly supported even though the other BOMs are detected, reported via get_bomid, and stripped from the data.
#[para]GB18030 falls back to cp936/gbk (unless a gb18030 encoding has been installed). Use -encoding iso8859-1 if this isn't suitable and you need to do your own processing of the bytes.
@ -1285,10 +1288,10 @@ namespace eval punk::fileline {
# -- --- --- ---
if {$opt_file ne ""} {
set filename $opt_file
set fd [open $filename r]
fconfigure $fd -translation binary -encoding $opt_translation;#should use translation binary to get actual line-endings - but we allow caller to override
#Always read encoding in binary - check for bom below and/or apply chosen opt_encoding
set filename $opt_file
set fd [open $filename r]
chan configure $fd -translation binary -encoding $opt_translation;#should use translation binary to get actual line-endings - but we allow caller to override
#Always read encoding in binary - check for bom below and/or apply chosen opt_encoding
set rawchunk [read $fd]
close $fd
if {[llength $values]} {
@ -1335,7 +1338,7 @@ namespace eval punk::fileline {
set is_reliabletxt 1
set startdata 4
} elseif {$maybe_bom eq "fffe0000"} {
#Technically ambiguous - could be utf-16le bom followed by utf-16 null character (2 byte null)
#Technically ambiguous - could be utf-16le bom followed by utf-16 null character (2 byte null)
puts stderr "WARNING - ambiguous BOM fffe0000 found. Treating as utf-32le - but could be utf-16le - consider manually setting -encoding or converting data to another encoding."
set bomid utf-32le
set bomenc utf-32le
@ -1360,7 +1363,7 @@ namespace eval punk::fileline {
set bomenc "binary" ;# utf-8???
set startdata 3
} elseif {$maybe_bom eq "84319533"} {
if {![dict exists [punk::char::page_names_dict gb18030]]} {
if {![dict exists [punk::char::page_names_dict gb18030] gb18030]} {
puts stderr "WARNING - no direct support for GB18030 (chinese) - falling back to cp936/gbk"
set bomenc cp936
} else {
@ -1374,7 +1377,7 @@ namespace eval punk::fileline {
set bomenc binary
set startdata 3
} elseif {[string match "2b2f76*" $maybe_bom]} {
puts stderr "WARNING utf-7 BOM 2b2f76 found - not supported. Falling back to binary and leaving BOM in data!"
puts stderr "WARNING utf-7 BOM 2b2f76 found - not supported. Falling back to binary and leaving BOM in data!"
#review - work out how to strip bom - last 2 bits of 4th byte belong to following character
set bomid utf-7
set bomenc binary
@ -1433,7 +1436,7 @@ namespace eval punk::fileline {
} else {
set datachunk [encoding convertfrom $bomenc [string range $rawchunk $startdata end]]
set encoding_selected $bomenc
}
}
} else {
#tcl 8.7 plus has utf-16le etc
set datachunk [encoding convertfrom $bomenc [string range $rawchunk $startdata end]]
@ -1443,7 +1446,7 @@ namespace eval punk::fileline {
#!?
if {$bomenc eq "binary"} {
set datachunk [string range $rawchunk $startdata end]
set encoding_selected binary
set encoding_selected binary
} else {
set datachunk [encoding convertfrom utf-8 [string range $rawchunk $startdata end]]
set encoding_selected utf-8
@ -1485,7 +1488,7 @@ namespace eval punk::fileline {
proc file_boundary_display {filename startbyte endbyte chunksize args} {
set fd [open $filename r] ;#use default error if file not readable
fconfigure $fd -translation binary
chan configure $fd -translation binary
set rawfiledata [read $fd]
close $fd
set textobj [class::textinfo new $rawfiledata]
@ -1510,7 +1513,7 @@ namespace eval punk::fileline::lib {
namespace path [namespace parent]
#*** !doctools
#[subsection {Namespace punk::fileline::lib}]
#[para] Secondary functions that are part of the API
#[para] Secondary functions that are part of the API
#[list_begin definitions]
@ -1532,12 +1535,12 @@ namespace eval punk::fileline::lib {
#[para]e.g
#[example_begin]
# range_spans_chunk_boundaries 10 1750 512
# is_span 1 boundaries {512 1024 1536}
# is_span 1 boundaries {512 1024 1536}
#[example_end]
#[para]The -offset <int> option
#[para]The -offset <int> option
#[example_begin]
# range_spans_chunk_boundaries 10 1750 512 -offset 2
# is_span 1 boundaries {514 1026 1538}
# is_span 1 boundaries {514 1026 1538}
#[example_end]
#[para] This function automatically uses lseq (if Tcl >= 8.7) when number of boundaries spanned is approximately greater than 75
if {[catch {package require Tcl 8.7-}]} {
@ -1576,12 +1579,12 @@ namespace eval punk::fileline::lib {
namespace eval punk::fileline::system {
#*** !doctools
#[subsection {Namespace punk::fileline::system}]
#[para] Internal functions that are not part of the API
#[para] Internal functions that are not part of the API
proc wordswap16 {data} {
#scan in one endianness - format in the other. Whether we scan le/be first doesn't matter as long as we format using the opposite endianness
binary scan $data s* elements ;#scan little endian
return [binary format S* $elements] ;#format big endian
return [binary format S* $elements] ;#format big endian
}
proc wordswap32 {data} {
binary scan $data i* elements
@ -1622,7 +1625,7 @@ namespace eval punk::fileline::system {
set start [expr {$start + ($chunksize - $smod)}]
if {$start > $end} {
return [list is_span 0 boundaries {}]
}
}
}
set boundaries [lseq $start to $end $chunksize]
#offset can be negative
@ -1632,7 +1635,7 @@ namespace eval punk::fileline::system {
} else {
set overflow 0
}
set boundaries [lmap v $boundaries[unset boundaries] {expr {$v + $opt_offset}}]
set boundaries [lmap v $boundaries[unset boundaries] {expr {$v + $opt_offset}}]
if {$overflow} {
#we don't know how many overflowed..
set inrange [list]
@ -1668,7 +1671,7 @@ namespace eval punk::fileline::system {
set opt_offset [dict get $opts -offset]
# -- --- --- ---
set is_span 0
set is_span 0
set smod [expr {$start % $chunksize}]
if {$smod != 0} {
set start [expr {$start + ($chunksize - $smod)}]
@ -1681,7 +1684,7 @@ namespace eval punk::fileline::system {
set btrack $bstart
set boff [expr {$btrack + $opt_offset}] ;#must be growing even if start and offset are negative - as chunksize is at least 1
while {$boff < $start} {
incr btrack $chunksize
incr btrack $chunksize
set boff [expr {$btrack + $opt_offset}]
}
set bstart $btrack
@ -1689,9 +1692,9 @@ namespace eval punk::fileline::system {
set bstart $start
}
for {set b $bstart} {[set boff [expr {$b + $opt_offset}]] <= $end} {incr b $chunksize} {
lappend boundaries $boff
}
lappend boundaries $boff
}
return [list is_span [expr {[llength $boundaries]>0}] boundaries $boundaries offset $opt_offset]
}
@ -1707,7 +1710,7 @@ namespace eval punk::fileline::ansi {
#*** !doctools
#[subsection {Namespace punk::fileline::ansi}]
#[para]These are ansi functions imported from punk::ansi - or no-ops if that package is unavailable
#[para]See [package punk::ansi] for documentation
#[para]See [package punk::ansi] for documentation
#[list_begin definitions]
variable enabled 1
#*** !doctools
@ -1720,11 +1723,11 @@ namespace eval punk::fileline::ansi {
}
# ++ +++ +++ +++ +++ +++ +++ +++ +++ +++ +++
## Ready
## Ready
package provide punk::fileline [namespace eval punk::fileline {
variable pkg punk::fileline
variable version
set version 999999.0a1.0
set version 999999.0a1.0
}]
return

30
src/modules/punk/icomm-999999.0a1.0.tm

@ -875,9 +875,9 @@ namespace eval ::punk::icomm {
![string equal $encoding $comm($chan,encoding)]} {
# This should not be entered yet
set comm($chan,encoding) $encoding
fconfigure $comm($chan,socket) -encoding $encoding
chan configure $comm($chan,socket) -encoding $encoding
foreach {i sock} [array get comm $chan,peers,*] {
fconfigure $sock -encoding $encoding
chan configure $sock -encoding $encoding
}
}
@ -935,10 +935,10 @@ namespace eval ::punk::icomm {
set nport [incr comm(lastport)]
}
set comm($chan,socket) $ret
fconfigure $ret -translation lf -encoding $comm($chan,encoding)
chan configure $ret -translation lf -encoding $comm($chan,encoding)
# If port was 0, system allocated it for us
set comm($chan,port) [lindex [fconfigure $ret -sockname] 2]
set comm($chan,port) [lindex [chan configure $ret -sockname] 2]
return ""
}
@ -1089,8 +1089,8 @@ namespace eval ::punk::icomm {
# coroutines to hide the CSP and properly handle everything
# event based.
fconfigure $fid -blocking 0
fileevent $fid readable [list ::punk::icomm::commIncomingOffered $chan $fid $addr $remport]
chan configure $fid -blocking 0
chan event $fid readable [list ::punk::icomm::commIncomingOffered $chan $fid $addr $remport]
return
}
@ -1111,8 +1111,8 @@ namespace eval ::punk::icomm {
# Protocol version line has been received, disable event handling
# again.
fileevent $fid readable {}
fconfigure $fid -blocking 1
chan event $fid readable {}
chan configure $fid -blocking 1
# a list of offered proto versions is the first word of first line
# remote id is the second word of first line
@ -1143,7 +1143,7 @@ namespace eval ::punk::icomm {
if {[dict exists $chanconf -sockname]} {
# If the remote host addr isn't our local host addr,
# then add it to the remote id.
if {[string equal [lindex [fconfigure $fid -sockname] 0] $addr]} {
if {[string equal [lindex [chan configure $fid -sockname] 0] $addr]} {
set id $remid
} else {
set id [list $remid $addr]
@ -1215,8 +1215,8 @@ namespace eval ::punk::icomm {
set comm($chan,peers,$id) $fid
}
set comm($chan,fids,$fid) $id
fconfigure $fid -translation lf -encoding $comm($chan,encoding) -blocking 0
fileevent $fid readable [list ::punk::icomm::commCollect $chan $fid]
chan configure $fid -translation lf -encoding $comm($chan,encoding) -blocking 0
chan event $fid readable [list ::punk::icomm::commCollect $chan $fid]
}
# ::punk::icomm::commLostConn --
@ -1324,7 +1324,7 @@ namespace eval ::punk::icomm {
# ::punk::icomm::commCollect --
#
# Internal command. Called from the fileevent to read from fid
# Internal command. Called from the chan event to read from fid
# and append to the buffer. This continues until we get a whole
# command, which we then invoke.
#
@ -1343,9 +1343,9 @@ namespace eval ::punk::icomm {
if {[catch {read $fid} nbuf] || [eof $fid]} {
commDebug {puts stderr "<$chan> collect/lost eof $fid = [eof $fid]"}
commDebug {puts stderr "<$chan> collect/lost nbuf = <$nbuf>"}
commDebug {puts stderr "<$chan> collect/lost [fconfigure $fid]"}
commDebug {puts stderr "<$chan> collect/lost [chan configure $fid]"}
fileevent $fid readable {} ;# be safe
chan event $fid readable {} ;# be safe
commLostConn $chan $fid "target application died or connection lost"
return
}
@ -1995,7 +1995,7 @@ proc ::punk::icomm::initlocal {{tcpport 0}} {
if {[string equal macintosh $::tcl_platform(platform)]} {
::punk::icomm::comm new ::punk::icomm::comm -port 0 -local 0 -listen 1
set ::punk::icomm::comm(localhost) \
[lindex [fconfigure $::punk::icomm::comm(::punk::icomm::comm,socket) -sockname] 0]
[lindex [chan configure $::punk::icomm::comm(::punk::icomm::comm,socket) -sockname] 0]
::punk::icomm::comm config -local 1
} else {
::punk::icomm::comm new ::punk::icomm::comm -port 0 -local 1 -listen 1

128
src/modules/punk/imap4-999999.0a1.0.tm

@ -234,7 +234,7 @@ tcl::namespace::eval punk::imap4::system {
if {$tag eq "*"} {
return [dict get $conlog $chan]
} else {
#retrieve
#retrieve
set loglist [dict get $conlog $chan]
#review - the relevant loglines should all be tagged with the 'request' key even if response line was a *
return [lsearch -all -inline -index 3 $loglist $tag]
@ -503,7 +503,7 @@ tcl::namespace::eval punk::imap4::proto {
+ {
if {$lastcmd eq "IDLE"} {
#todo - verify '+ idling' case?
set info($chan,idle) [clock seconds]
set info($chan,idle) [clock seconds]
} else {
#assert - can't happen
}
@ -558,11 +558,11 @@ tcl::namespace::eval punk::imap4::proto {
append line $buf
# Check if there is a literal specified.
# It will always occur at the end of a line - followed by the data to read
# It will always occur at the end of a line - followed by the data to read
if {[regexp {{([0-9]+)}\s*$} $buf => length]} {
# puts "Reading $length bytes of literal..."
set chunk [read $chan $length]
lappend literals $chunk
lappend literals $chunk
#add_conlog $chan $side $type <datalist>
::punk::imap4::system::add_conlog $chan s $request_tag literal [list [dict create length $length lines [llength [split $chunk \n]]]]
if {[dict get $coninfo $chan debug]} {
@ -570,7 +570,7 @@ tcl::namespace::eval punk::imap4::proto {
::punk::imap4::system::add_conlog $chan s $request_tag chunk [list [list length $length chunk $chunk]]
}
} else {
#We are at the end of a single line,
#We are at the end of a single line,
#or a sequence of 1 or more lines which had trailing literal specifiers {nnn} followed by data we have read.
break
}
@ -667,7 +667,7 @@ tcl::namespace::eval punk::imap4::proto {
#If tag eq * - we could still have an OK not stripped from line above
#e.g initial connection response
#REVIEW -
#REVIEW -
if {!$dirty && $tag eq {*}} {
switch -regexp -nocase -- $line {
{^[0-9]+\s+EXISTS} {
@ -699,7 +699,7 @@ tcl::namespace::eval punk::imap4::proto {
}
{^METADATA} {
#e.g
#* METADATA test1 ("/private/specialuse" NIL)
#* METADATA test1 ("/private/specialuse" NIL)
# or
#* METADATA Drafts ("/private/specialuse" {7}
# \Drafts
@ -989,10 +989,10 @@ tcl::namespace::eval punk::imap4::proto {
# "HEADER.FIELD", "\Answered", "$Forwarded"
#set pattern {([\w\.]+\[[^\[]+\]|[\w\.]+|[\\\$]\w+)}
#some examples that should also match:
# BODY[]
# BODY[]
# BODY[]<0.100> ;#first 100 bytes
# BINARY.PEEK[1]<100.200>
set pattern {([\w\.]+\[[^\[]*\](?:\<[^\>]*\>)*|[\w\.]+|[\\\$]\w+)}
set pattern {([\w\.]+\[[^\[]*\](?:\<[^\>]*\>)*|[\w\.]+|[\\\$]\w+)}
if {![regexp $pattern $data => match]} {
protoerror $chan "IMAP data format error: '$data'"
}
@ -1218,11 +1218,11 @@ tcl::namespace::eval punk::imap4 {
"Connection security.
TLS/SSL is recommended (implicit TLS).
If port is 143 and -security is omitted, then it will
If port is 143 and -security is omitted, then it will
default to STARTTLS.
For any other port, or omitted port, the default for
-security is TLS/SSL.
ie if no channel security is wanted, then -security
ie if no channel security is wanted, then -security
should be explicitly set to None."
@values -min 1 -max 2
hostname -optional 0 -help\
@ -1237,7 +1237,7 @@ tcl::namespace::eval punk::imap4 {
port -optional 1 -type integer -help\
"Port to connect to.
If port is omitted:
defaults to 143 when -security None or STARTTLS
defaults to 143 when -security None or STARTTLS
defaults to 993 when -security TLS/SSL or -security is omitted."
}]
proc OPEN {args} {
@ -1276,11 +1276,11 @@ tcl::namespace::eval punk::imap4 {
}
}
} else {
#port is specified and not 0
set port $specified_port
#port is specified and not 0
set port $specified_port
if {$port == 143} {
if {$opt_security eq "unspecified"} {
set opt_security STARTTLS
set opt_security STARTTLS
}
} else {
#assume any other port is TLS/SSL by default if user didn't specify
@ -1294,7 +1294,7 @@ tcl::namespace::eval punk::imap4 {
upvar ::punk::imap4::proto::info info
upvar ::punk::imap4::proto::coninfo coninfo
#variable use_ssl
#variable use_ssl
if {$opt_debug} {
puts "I: open $address $port (SECURITY=$opt_security)"
}
@ -1312,7 +1312,7 @@ tcl::namespace::eval punk::imap4 {
# set chan [twapi::starttls $insecure_chan -peersubject mail.11email.com]
# set connected 1
#}
if {!$connected} {
if {!$connected} {
catch {package require tls} ;#review
if {[info procs ::tls::socket] eq ""} {
error "Package TLS must be loaded for STARTTLS connections."
@ -1329,7 +1329,7 @@ tcl::namespace::eval punk::imap4 {
set chan $insecure_chan; #upgraded
#processline $chan
puts "--> [lastline $chan]"
#get new caps response?
#get new caps response?
return $chan
} else {
puts stderr "STARTTLS failed"
@ -1345,7 +1345,7 @@ tcl::namespace::eval punk::imap4 {
#implicit TLS - preferred
set chan [::tls::socket $address $port]
}
}
}
chan configure $chan -translation binary
dict set coninfo $chan [dict create hostname $address port $port debug $opt_debug security $opt_security]
@ -1392,22 +1392,22 @@ tcl::namespace::eval punk::imap4 {
# is known as STARTTLS.
# (implicit TLS on a dedicated port is the modern preference,
# but this should be supported in the client API even if many servers
# move away from it)
# move away from it)
proc STARTTLS {chan} {
package require tls
#puts "Starting TLS"
#puts "Starting TLS"
punk::imap4::proto::requirecaps $chan STARTTLS
set clitag [punk::imap4::proto::request $chan STARTTLS]
if {[punk::imap4::proto::getresponse $chan $clitag] != 0} {
#puts "error sending STARTTLS"
return 1
}
#puts "TLS import"
set chan [::tls::import $chan]
#puts "TLS handshake"
#tls::handshake
#returns 0 if handshake still in progress (non-blocking)
#returns 1 if handshake was successful
@ -1509,7 +1509,7 @@ tcl::namespace::eval punk::imap4 {
}
}
}
append result
append result
}
return $result
}
@ -1521,10 +1521,10 @@ tcl::namespace::eval punk::imap4 {
#some headers have multipl values (SMTP traces)
#also consider the somewhat contrived use of partials:
# FETCH (BODY[]<0.100> BODY[]<0.10>)
#These are returned in the FETCH response as "BODY[]<0> {100}" and "BODY[]<0> {10}"
#These are returned in the FETCH response as "BODY[]<0> {100}" and "BODY[]<0> {10}"
#This results in us having a msginfo key of "BODY[]<0>" with 2 values.
#
proc _set_msginfo_field {chan msgnum request_tag field value} {
variable msginfo
if {![dict exists $msginfo $chan $msgnum]} {
@ -1533,22 +1533,22 @@ tcl::namespace::eval punk::imap4 {
set msgdata [dict get $msginfo $chan $msgnum]
}
if {![dict exists $msgdata $field]} {
set fieldinfo [dict create count 1 values [list $value] request $request_tag]
set fieldinfo [dict create count 1 values [list $value] request $request_tag]
} else {
#update field info for msgnum
set prev_fieldinfo [dict get $msgdata $field]
set prev_request [dict get $prev_fieldinfo request]
set prev_request [dict get $prev_fieldinfo request]
if {$prev_request ne $request_tag} {
#new request - can overwrite
set fieldinfo [dict create count 1 values [list $value] request $request_tag]
} else {
#same request - duplicate header/field e.g Received: header - we need to store all.
set fieldinfo $prev_fieldinfo
set fieldinfo $prev_fieldinfo
dict incr fieldinfo count
dict lappend fieldinfo values $value
}
}
dict set msgdata $field $fieldinfo
dict set msgdata $field $fieldinfo
dict set msginfo $chan $msgnum $msgdata
#set msginfo($chan,$msgnum,$field) $value
}
@ -1570,7 +1570,7 @@ tcl::namespace::eval punk::imap4 {
#no change to count or request fields
dict set fieldinfo values $values
dict set msginfo $chan $msgnum $field $fieldinfo
dict set msginfo $chan $msgnum $field $fieldinfo
#append msginfo($chan,$msgnum,$field) $value
}
@ -1585,8 +1585,8 @@ tcl::namespace::eval punk::imap4 {
for {set i 0} {$i < $count} {incr i} {
append out "$msgseq $prop [lindex [dict get $propdata values] $i]"
}
}
}
}
}
return $out
}
@ -1603,14 +1603,14 @@ tcl::namespace::eval punk::imap4 {
"Login using the IMAP LOGIN command.
"
@leaders -min 1 -max 1
chan -optional 0
chan -optional 0
@opts
-ignorestate -type none -help\
"Send the LOGIN even if protocol state is not appropriate"
-ignorelogindisabled -type none -help\
"Ignore the LOGINDISABLED capability
from the server and send LOGIN anyway.
(There should be no need to use this
(There should be no need to use this
except for server testing purposes)"
@values -min 2 -max 2
username
@ -1633,7 +1633,7 @@ tcl::namespace::eval punk::imap4 {
}
}
if {!$opt_ignorestate} {
punk::imap4::proto::requirestate $chan NOAUTH
punk::imap4::proto::requirestate $chan NOAUTH
}
set rtag [punk::imap4::proto::request $chan "LOGIN $username $password"]
if {[punk::imap4::proto::getresponse $chan $rtag] != 0} {
@ -1647,7 +1647,7 @@ tcl::namespace::eval punk::imap4 {
@id -id ::punk::imap4::AUTH_PLAIN
@cmd -name punk::imap4::AUTH_PLAIN -help\
"PLAIN SASL Authentication mechanism.
This uses the 'initial response' to send
the base64 encoded authzn authn password
in the same line as AUTHENTICATE PLAIN.
@ -1657,17 +1657,17 @@ tcl::namespace::eval punk::imap4 {
and the client sends the credentials after
getting a continuation (+) from the server."
@leaders -min 1 -max 1
chan -optional 0
chan -optional 0
@opts
-ignorestate -type none -help\
"Send the AUTHENTICATE even if protocol state is not appropriate"
-authorization -type string -default "" -help\
"authorization identity (identity to act as)
Usually it is not necessary to provide an
Usually it is not necessary to provide an
authorization identity - as it will be derived
from the credentials. ie from the
from the credentials. ie from the
'authentication identity' which is the username.
"
"
@values -min 2 -max 2
username -help\
"Authentication identity"
@ -1683,7 +1683,7 @@ tcl::namespace::eval punk::imap4 {
if {$opt_ignorestate} {
set allowstates *
} else {
set allowstates NOAUTH
set allowstates NOAUTH
}
set username [dict get $values username]
set password [dict get $values password]
@ -1738,7 +1738,7 @@ tcl::namespace::eval punk::imap4 {
set rtag [punk::imap4::proto::request $chan "$cmd $mailbox"]
if {[punk::imap4::proto::getresponse $chan $rtag] != 0} {
#array set mboxinfo $savedmboxinfo
set info($chan,state) AUTH
set info($chan,state) AUTH
return 1
}
@ -1869,7 +1869,7 @@ tcl::namespace::eval punk::imap4 {
#todo "$" data-item ?
foreach data_item $query_items {
set DATA_ITEM [string toupper $data_item]
set DATA_ITEM [string toupper $data_item]
switch -- $DATA_ITEM {
ALL - FAST - FULL {lappend items $DATA_ITEM}
BODY -
@ -1974,7 +1974,7 @@ tcl::namespace::eval punk::imap4 {
#based on assumed simple value queries such as specific properties and headers that are individually specified.
set fetchresult [dict create]
for {set i $start} {$i <= $end} {incr i} {
set flagdict [dict get $msginfo $chan $i]
set flagdict [dict get $msginfo $chan $i]
#extract the fields that were added for this request_tag only
dict for {f finfo} $flagdict {
if {[dict get $finfo request] eq $request_tag} {
@ -1988,7 +1988,7 @@ tcl::namespace::eval punk::imap4 {
#return $mailinfo
set mailinfo {}
set fields [list]
set fields [list]
#todo - something better
foreach itm $items {
if {$itm ni {ALL FAST FULL}} {
@ -1998,7 +1998,7 @@ tcl::namespace::eval punk::imap4 {
#lappend fields {*}$hdrfields
set fields [list {*}$fields {*}$hdrfields]
for {set i $start} {$i <= $end} {incr i} {
set mailrec [list]
set mailrec [list]
foreach {f} $fields {
#lappend mailrec [msginfo $chan $i $f ""]
set finfo [msginfo $chan $i $f ""]
@ -2144,7 +2144,7 @@ tcl::namespace::eval punk::imap4 {
The cached results can be checked with
the punk::imap4::has_capability command."
@leaders -min 1 -max 1
chan -optional 0
chan -optional 0
@opts
@values -min 0 -max 0
}]
@ -2176,7 +2176,7 @@ tcl::namespace::eval punk::imap4 {
autologout timer on the server.
"
@leaders -min 1 -max 1
chan -optional 0
chan -optional 0
@opts
@values -min 0 -max 0
}]
@ -2201,7 +2201,7 @@ tcl::namespace::eval punk::imap4 {
return 1
}
#array set mboxinfo {} ;#JMN
#array set mboxinfo {} ;#JMN
set mboxinfo [dict create]
set info($chan,state) AUTH
return 0
@ -2233,7 +2233,7 @@ tcl::namespace::eval punk::imap4 {
see also RFC3691 - IMAP UNSELECT command
"
@leaders -min 1 -max 1
chan -optional 0
chan -optional 0
@opts
-ignorestate -type none -help\
"Send the UNSELECT even if protocol state is not appropriate"
@ -2260,14 +2260,14 @@ tcl::namespace::eval punk::imap4 {
if {[punk::imap4::proto::simplecmd $chan UNSELECT {*}$allowstates {}]} {
return 1
}
#array set mboxinfo {} ;#JMN
#array set mboxinfo {} ;#JMN
set mboxinfo [dict create]
set info($chan,state) AUTH
return 0
}
proc NAMESPACE {chan} {
punk::imap4::proto::simplecmd $chan NAMESPACE *
punk::imap4::proto::simplecmd $chan NAMESPACE *
}
# Create a new mailbox.
@ -2293,7 +2293,7 @@ tcl::namespace::eval punk::imap4 {
#S: * METADATA "Foldername" (/private/specialuse {5}
#S: \Junk
#S: )
#S: <tag> OK Completed
#S: <tag> OK Completed
set annotation [string trim $annotation]
if {![string match "/private/?*" $annotation] && ![string match "/shared/?*" $annotation]} {
error "GETMETADATA annotation must begin with /shared/ or /private/"
@ -2306,10 +2306,10 @@ tcl::namespace::eval punk::imap4 {
@cmd -name "punk::imap4::SETMETDATA" -help\
"Set metadata on mailbox"
@leaders -min 1 -max 1
chan
chan
@opts
@values -min 3 -max 3
mailbox
mailbox
annotation -choicerestricted 0 -choices {
/private/specialuse /private/squat /private/sieve /private/sharedseen /private/comment
/private/expire /private/news2mail /private/pop3showafter
@ -2363,7 +2363,7 @@ tcl::namespace::eval punk::imap4 {
#TODO
proc IDLE {chan} {
if {[punk::imap4::prot::has_capability $chan IDLE]} {
punk::imap4::proto::simplecmd $chan IDLE {AUTH SELECT}
punk::imap4::proto::simplecmd $chan IDLE {AUTH SELECT}
} else {
error "IMAP SERVER has NOT advertised the capability IDLE."
}
@ -2390,9 +2390,9 @@ tcl::namespace::eval punk::imap4 {
@cmd -name "punk::imap4::FOLDERS" -help\
"List of folders"
@leaders -min 1 -max 1
chan
chan
@opts
-ignorestate -type none
-ignorestate -type none
-inline -type none
@values -min 0 -max 2
ref -default ""
@ -2498,10 +2498,10 @@ tcl::namespace::eval punk::imap4 {
"Debug mode.
This is a developer mode that provides a basic REPL
(Read Eval Print Loop) to interact more directly with the
server.
server.
Every line entered is sent verbatim to the
server (after the automatic addition of the request identifier/tag).
It's possible to execute Tcl commands by starting the line
with a forward slash."
@leaders -min 0 -max 0
@ -2542,7 +2542,7 @@ tcl::namespace::eval punk::imap4 {
puts $l
}
set prev_chan_debug [dict get $coninfo $chan debug]
set prev_chan_debug [dict get $coninfo $chan debug]
dict set coninfo $chan debug 1 ;#ensure debug for this chan on while in debugmode
@ -2559,7 +2559,7 @@ tcl::namespace::eval punk::imap4 {
gets stdin line
if {![string length $line]} continue
if {$line eq {!}} {
break
break
}
switch -glob -- $line {
info {
@ -3260,7 +3260,7 @@ tcl::namespace::eval punk::imap4 {
lappend PUNKARGS [list {
@id -id "(package)punk::imap4"
@package -name "punk::imap4" -help\
"Package
"Package
Description"
}]

574
src/modules/punk/lib-999999.0a1.0.tm

File diff suppressed because it is too large Load Diff

100
src/modules/punk/mix/base-0.1.tm

@ -18,7 +18,7 @@ namespace eval punk::mix::base {
set extension ""
}
#---------
uplevel #0 [list interp alias {} $cmdname {} punk::mix::base::_cli -extension $extension]
}
proc _cli {args} {
@ -69,7 +69,7 @@ namespace eval punk::mix::base {
}
#puts stderr "arglen:[llength $args]"
#puts stdout "_unknown '$ns' '$args'"
set d_commands [get_commands -extension $extension]
set all_commands [list {*}[dict get $d_commands main] {*}[dict get $d_commands base]]
@ -98,11 +98,11 @@ namespace eval punk::mix::base {
}
tailcall [namespace current] $subcommand {*}$argvals {*}$args -extension $from_ns
} else {
if {[regexp {.*[*?].*} $subcommand]} {
if {[regexp {.*[*?].*} $subcommand]} {
set d_commands [get_commands -extension $from_ns]
set all_commands [list {*}[dict get $d_commands main] {*}[dict get $d_commands base]]
set matched_commands [lsearch -all -inline $all_commands $subcommand]
set commands ""
set commands ""
foreach m $matched_commands {
append commands $m \n
}
@ -113,12 +113,12 @@ namespace eval punk::mix::base {
}
proc _split_args {arglist} {
#don't assume arglist is fully paired.
set posn [lsearch $arglist -extension]
set posn [lsearch $arglist -extension]
set opts [list]
if {$posn >= 0} {
if {$posn+2 <= [llength $arglist]} {
set opts [list -extension [lindex $arglist $posn+1]]
set argsremaining [lreplace $arglist $posn $posn+1]
set argsremaining [lreplace $arglist $posn $posn+1]
} else {
#no value supplied to -extension
error "punk::mix::base::_split_args - no value found for option '-extension'. Supply a value or omit the option."
@ -151,7 +151,7 @@ namespace eval punk::mix::base {
if {![string length $extension]} {
set extension [namespace qualifiers [lindex [info level -1] 0]]
}
set maincommands [list]
#extension may still be blank e.g if punk::mix::base::get_commands called directly
if {[string length $extension]} {
@ -164,7 +164,7 @@ namespace eval punk::mix::base {
}
foreach c $nscommands {
set cmd [namespace tail $c]
lappend maincommands $cmd
lappend maincommands $cmd
}
set maincommands [lsort $maincommands]
}
@ -190,29 +190,29 @@ namespace eval punk::mix::base {
set basecommands [lsort $basecommands]
return [list main $maincommands base $basecommands]
return [list main $maincommands base $basecommands]
}
proc help {args} {
#' **%ensemblecommand% help** *args*
#'
#'
#' Help for ensemble commands in the command line interface
#'
#'
#'
#'
#' Arguments:
#'
#'
#' * args - first word of args is the helptopic requested - usually a command name
#' - calling help with no arguments will list available commands
#'
#'
#' Returns: help text (text)
#'
#'
#' Examples:
#'
#'
#' ```
#' %ensemblecommand% help <commandname>
#' ```
#'
#'
#'
#'
#extension.= @@opts/@?@-extension,args@@args=>. [_split_args $args] {|
# >} inspect -label a {|
@ -220,7 +220,7 @@ namespace eval punk::mix::base {
# pipecase ,0/1/#= $switchargs {|
# e/0
# >} .=>. {set e}
# pipecase /1,1/1/#= $switchargs
# pipecase /1,1/1/#= $switchargs
#} |@@ok/result> <e/0| [namespace qualifiers [lindex [info level -1] 0]]
@ -242,13 +242,13 @@ namespace eval punk::mix::base {
#puts stderr "-1:[info level -1]"
set command_info [punk::mix::base::get_commands -extension $extension]
set subhelp1 [lindex $args 0]
set subhelp1 [lindex $args 0]
if {[string length $subhelp1]} {
if {[regexp {[*?]} $subhelp1]} {
set helpstr ""
append helpstr "matched commands:\n"
dict for {source cmdlist} $command_info {
set matches [lsearch -all -inline -glob $cmdlist $subhelp1]
set matches [lsearch -all -inline -glob $cmdlist $subhelp1]
if {[llength $matches]} {
append helpstr \n " $source"
foreach cmd $matches {
@ -271,7 +271,7 @@ namespace eval punk::mix::base {
} else {
set a [interp alias {} ${ns}::$subhelp1]
if {[string length $a]} {
return "alias: $subhelp1 target: $a"
return "alias: $subhelp1 target: $a"
} else {
return "command: $subhelp1 (No info available)"
}
@ -298,7 +298,7 @@ namespace eval punk::mix::base {
return $helpstr
}
#proc dostuff {args} {
# extension@@opts/@?@-extension,args@@args= [_split_args $args]
# extension@@opts/@?@-extension,args@@args= [_split_args $args]
# puts stdout "base doingstuff-with-args:'$args'-in-namespace:'[namespace current]'"
#}
namespace eval lib {
@ -338,7 +338,7 @@ namespace eval punk::mix::base {
if {![string length [set candidate [punk::repo::find_candidate $path]]]} {
error "find_source_module_paths cannot determine a suitable project root at or above path '$path' - path supplied should be within a project"
}
#we can return module paths even if the project isn't yet under revision control
#we can return module paths even if the project isn't yet under revision control
set src_subs [glob -nocomplain -dir [file join $candidate src] -type d -tail *]
set antipatterns [list *.vfs vendor* lib _build doc embedded runtime bootsupport]
set tm_folders [list]
@ -346,8 +346,8 @@ namespace eval punk::mix::base {
set is_ok 1
foreach anti $antipatterns {
if {[string match $anti $sub]} {
set is_ok 0
break
set is_ok 0
break
}
}
if {!$is_ok} {
@ -363,7 +363,7 @@ namespace eval punk::mix::base {
#set podfolders [glob -nocomplain -dir $testfolder -type d -tail #modpod-*]
if {[llength [glob -nocomplain -dir $testfolder -type f -tail *.tm]] || [llength [glob -nocomplain -dir $testfolder -type d -tail #modpod-*]]} {
lappend tm_folders $testfolder
}
}
}
return $tm_folders
}
@ -420,7 +420,7 @@ namespace eval punk::mix::base {
}
#todo - move cksum stuff to punkcheck - more logical home
proc cksum_path_content {path args} {
dict set args -cksum_content 1
@ -440,7 +440,7 @@ namespace eval punk::mix::base {
# - try builtin zlib crc instead?
#sha1 is performant - and this is not being used in a cryptographic or adversarial context - so performance and practical unlikelihood of accidental collisions should be the main consideration.
#adler32 is fastest for some larger files of a few MB but slower on small files (possibly due to Tcl-based file load?)
#sha1 as at 2023 seems a reasonable default
#sha1 as at 2023 seems a reasonable default
proc cksum_algorithms {} {
variable sha3_implementation
#sha2 is an alias for sha256
@ -448,11 +448,11 @@ namespace eval punk::mix::base {
set algs [list md5 sha1 sha2 sha256 cksum adler32]
set sha3_algs [list sha3 sha3-224 sha3-256 sha3-384 sha3-512]
if {[auto_execok sqlite3] ne ""} {
lappend algs {*}$sha3_algs
lappend algs {*}$sha3_algs
set sha3_implementation sqlite3_sha3
} else {
if {[auto_execok fossil] ne ""} {
lappend algs {*}$sha3_algs
lappend algs {*}$sha3_algs
set sha3_implementation fossil_sha3
}
}
@ -506,7 +506,7 @@ namespace eval punk::mix::base {
}
set base [file dirname $path]
set startdir [pwd]
set defaults [cksum_default_opts]
set known_opts [dict keys $defaults]
foreach {k v} $args {
@ -521,7 +521,7 @@ namespace eval punk::mix::base {
#if {![file exists $path]} {
# return [list cksum "" opts $opts]
#}
if {[catch {file type $path} ftype]} {
return [list cksum "<PATHNOTFOUND>" opts $opts]
}
@ -620,7 +620,7 @@ namespace eval punk::mix::base {
if {$path eq $base} {
#attempting to cksum at root/volume level of a filesystem.. extra work
#This needs fixing for general use.. not necessarily just for project repos
#This needs fixing for general use.. not necessarily just for project repos
puts stderr "cksum_path doesn't yet support cksum of entire volume. (todo)"
return [list error unsupported_path opts $opts]
}
@ -671,8 +671,8 @@ namespace eval punk::mix::base {
set archivename $tmplocation/[punk::mix::util::tmpfile].tar
cd $base ;#cd is process-wide.. keep cd in effect for as small a scope as possible. (review for thread issues)
#temp emission to stdout.. todo - repl telemetry channel
#temp emission to stdout.. todo - repl telemetry channel
puts stdout "cksum_path: creating temporary tar archive for $path"
puts -nonewline stdout " at: $archivename ..."
set tsstart [clock millis]
@ -692,7 +692,7 @@ namespace eval punk::mix::base {
set ms [expr {$tsend - $tsstart}]
puts stdout " tar::create done ($ms ms)"
puts stdout " NOTE: install tar executable for potentially *much* faster directory checksum processing"
}
}
if {$ftype eq "file"} {
set sizeinfo "(size [punk::lib::format_number [file size $target]] bytes)"
@ -718,7 +718,7 @@ namespace eval punk::mix::base {
set cksum [{*}$cksum_command $path]
}
} else {
error "cksum_path unsupported $opts for path type [file type $path]"
error "cksum_path unsupported $opts for path type [file type $path]"
}
}
set result [dict create]
@ -733,7 +733,7 @@ namespace eval punk::mix::base {
#base can be empty string in which case paths must be absolute
#expect dict_path_cksum to be a dict keyed on relpath where each value is a dictionary with keys cksum and opts
# ie subdict for <path> can be created from output of cksum_path <path> (for already known values not requiring filling)
# or cksum "" opts [cksum_default_opts] or cksum "" opts {} (for cksum to be filled using supplied cksum opts if any)
# or cksum "" opts [cksum_default_opts] or cksum "" opts {} (for cksum to be filled using supplied cksum opts if any)
proc fill_relativecksums_from_base_and_relativepathdict {base {dict_path_cksum {}}} {
if {$base eq ""} {
set error_paths [list]
@ -775,7 +775,7 @@ namespace eval punk::mix::base {
}
}
if {$base ne ""} {
set fullpath [file join $base $path]
set fullpath [file join $base $path]
} else {
set fullpath $path
}
@ -820,7 +820,7 @@ namespace eval punk::mix::base {
#Here we will raise an error if cksum exists and is not empty or a tag - whereas the multiple path version will honour valid-looking prefilled cksum values (ie will pass them through)
#base is the presumed location to store the checksum file. The caller should retain (normalize if relative)
proc get_relativecksum_from_base {base specifiedpath args} {
if {$base ne ""} {
if {$base ne ""} {
#targetpath ideally should be within same project tree as base if base supplied - but not necessarily below it
#we don't necessarily want to restrict this to use in punk projects though - so we'll allow anything with a common prefix
if {[file pathtype $specifiedpath] eq "relative"} {
@ -846,9 +846,9 @@ namespace eval punk::mix::base {
#absolute base with no shared prefix doesn't make sense - we could ignore it - but better to error-out and require the caller specify an empty base
error "get_relativecksum_from_base error: base '$base' and specifiedpath '$specifiedpath' don't share a common root. Use empty-string for base if independent absolute path is required"
}
set targetpath $specifiedpath
set targetpath $specifiedpath
set storedpath [punk::path::relative $base $specifiedpath]
}
} else {
if {[file type $specifiedpath] eq "relative"} {
@ -863,7 +863,7 @@ namespace eval punk::mix::base {
#
#NOTE: specifiedpath can be a relative path (to cwd) when base is empty
#OR - a relative path when base itself is relative e.g base: somewhere targetpath somewhere/etc
#OR - a relative path when base itself is relative e.g base: somewhere targetpath somewhere/etc
#possibly also: base: somewhere targetpath: ../elsewhere/etc
#
#todo - write tests
@ -881,7 +881,7 @@ namespace eval punk::mix::base {
set ckopts [cksum_filter_opts {*}$args]
set ckinfo [cksum_path $targetpath {*}$ckopts]
set keyvals $args ;# REVIEW
dict set keyvals cksum [dict get $ckinfo cksum]
#dict set keyvals cksum_all_opts [dict get $ckinfo opts]
@ -891,7 +891,7 @@ namespace eval punk::mix::base {
}
#set relpath [punk::repo::path_strip_alreadynormalized_prefixdepth $fullpath $base] ;#empty base ok noop
#storedpath is relative if possible
#storedpath is relative if possible
return [dict create $storedpath $keyvals]
}
@ -910,7 +910,7 @@ namespace eval punk::mix::base {
dict set dict_cksums [file join $buildrelpath $vname.exe] [list cksum ""]
}
#buildruntime.exe obsolete..
#buildruntime.exe obsolete..
set fullpath_buildruntime $buildfolder/buildruntime.exe
set ckinfo_buildruntime [cksum_path $fullpath_buildruntime]
@ -944,7 +944,7 @@ namespace eval punk::mix::base {
}
proc get_all_build_cksums_stored {path} {
set buildfolder [get_build_workdir $path]
set vfscontainer [file dirname $buildfolder]
set vfslist [glob -nocomplain -dir $vfscontainer -type d -tail *.vfs]
set dict_cksums [dict create]
@ -963,7 +963,7 @@ namespace eval punk::mix::base {
}
set vfscontainer [file dirname $vfsfolder]
set buildfolder $vfscontainer/_build
set dict_vfs [get_vfs_build_cksums $vfsfolder]
set dict_vfs [get_vfs_build_cksums $vfsfolder]
set data ""
dict for {path cksum} $dict_vfs {
append data "$path $cksum" \n

118
src/modules/punk/mix/cli-999999.0a1.0.tm

@ -7,7 +7,7 @@
# (C) 2023
#
# @@ Meta Begin
# Application punk::mix::cli 999999.0a1.0
# Application punk::mix::cli 999999.0a1.0
# Meta platform tcl
# Meta license <unspecified>
# @@ Meta End
@ -19,7 +19,7 @@
##e.g package require frobz
package require punk::repo
package require punk::ansi
package require punkcheck ;#checksum and/or timestamp records
package require punkcheck ;#checksum and/or timestamp records
@ -33,7 +33,7 @@ namespace eval punk::mix::cli {
namespace ensemble create
variable initialised 0
#lazy _init - called by punk::mix::base::_cli when ensemble used
#lazy _init - called by punk::mix::base::_cli when ensemble used
proc _init {args} {
variable initialised
if {$initialised} {
@ -52,7 +52,7 @@ namespace eval punk::mix::cli {
catch {
package require punk::mix::commandset::project
punk::overlay::import_commandset project . ::punk::mix::commandset::project
punk::overlay::import_commandset projects . ::punk::mix::commandset::project::collection
punk::overlay::import_commandset projects . ::punk::mix::commandset::project::collection
}
if {[catch {
package require punk::mix::commandset::layout
@ -91,12 +91,12 @@ namespace eval punk::mix::cli {
}
proc stat {{workingdir ""} args} {
dict set args -v 0
punk::mix::cli::lib::get_status $workingdir {*}$args
dict set args -v 0
punk::mix::cli::lib::get_status $workingdir {*}$args
}
proc status {{workingdir ""} args} {
dict set args -v 1
punk::mix::cli::lib::get_status $workingdir {*}$args
dict set args -v 1
punk::mix::cli::lib::get_status $workingdir {*}$args
}
@ -128,13 +128,13 @@ namespace eval punk::mix::cli {
set project_base [punk::repo::find_candidate]
set sourcefolder $project_base/src
puts stderr "WARNING - project not under git or fossil control"
puts stderr "Using base folder $project_base"
puts stderr "Using base folder $project_base"
} else {
set sourcefolder $startdir
}
}
#review - why can't we be anywhere in the project?
#review - why can't we be anywhere in the project?
#also - if no make.tcl - can we use the running shell's make.tcl ? (after prompting user?)
if {([file tail $sourcefolder] ne "src") || (![file exists $sourcefolder/make.tcl])} {
puts stderr "dev make must be run from src folder containing make.tcl - unable to proceed (cwd: [pwd])"
@ -157,7 +157,7 @@ namespace eval punk::mix::cli {
if {![string length $project_base]} {
puts stderr "WARNING no git or fossil repository detected."
puts stderr "Using base folder $startdir"
puts stderr "Using base folder $startdir"
set project_base $startdir
}
@ -178,7 +178,7 @@ namespace eval punk::mix::cli {
}
}
#cd $sourcefolder
#use run so that stdout visible as it goes
if {![catch {run --timeout=55000 -debug [info nameofexecutable] $sourcefolder/make.tcl {*}$args} exitinfo]} {
#todo - notify if exit because of timeout!
@ -198,7 +198,7 @@ namespace eval punk::mix::cli {
puts stdout "OK make finished "
return true
}
}
}
proc Kettle {args} {
tailcall lib::kettle_call lib {*}$args
@ -241,7 +241,7 @@ namespace eval punk::mix::cli {
# -- --- --- --- --- --- --- --- --- --- --- --- --- ---
if {$opt_strict} {
if {[regexp {[A-Z]} $modulename]} {
error "$opt_errorprefix '$modulename' contains uppercase which is not recommended as per tip 590, and option -strict is set to 1"
error "$opt_errorprefix '$modulename' contains uppercase which is not recommended as per tip 590, and option -strict is set to 1"
}
}
@ -272,7 +272,7 @@ namespace eval punk::mix::cli {
} elseif {[regexp {[A-Z]} $modulename]} {
set msg "module names containing uppercase are not recommended (see tip 590).\n"
append msg "Please retype the module name '$modulename' to proceed.\n"
append msg "If you type it exactly as it was you will be allowed to proceed with uppercase anyway\n"
append msg "If you type it exactly as it was you will be allowed to proceed with uppercase anyway\n"
append msg "Retype it all in lowercase to use recommended naming"
set answer [util::askuser $msg]
if {[regexp {[A-Z]} $answer]} {
@ -285,11 +285,11 @@ namespace eval punk::mix::cli {
}
set modulename $answer
} else {
#user has resupplied modulename all as lowercase
#user has resupplied modulename all as lowercase
if {$answer eq [string tolower $modulename]} {
set finalised 1
} else {
#.. but it doesn't match original - require rerun
#.. but it doesn't match original - require rerun
}
set modulename $answer
}
@ -332,7 +332,7 @@ namespace eval punk::mix::cli {
if {[string first "::" $projectname] >= 0} {
error "$opt_errorprefix '$projectname' cannot contain namespace separator '::'"
}
return $projectname
return $projectname
}
proc validate_name_not_empty_or_spaced {name args} {
set opts [list\
@ -394,7 +394,7 @@ namespace eval punk::mix::cli {
set result ""
if {$workingdir ne ""} {
if {[file pathtype $workingdir] ne "absolute"} {
set workingdir [file normalize $workingdir]
set workingdir [file normalize $workingdir]
}
set active_dir $workingdir
} else {
@ -403,10 +403,10 @@ namespace eval punk::mix::cli {
set defaults [dict create\
-v 1\
]
set opts [dict merge $defaults $args]
# -- --- --- --- --- --- --- --- ---
set opts [dict merge $defaults $args]
# -- --- --- --- --- --- --- --- ---
set opt_v [dict get $opts -v]
# -- --- --- --- --- --- --- --- ---
# -- --- --- --- --- --- --- --- ---
set repopaths [punk::repo::find_repos [pwd]]
@ -417,7 +417,7 @@ namespace eval punk::mix::cli {
append result [dict get $repopaths warnings]
lassign [lindex $repos 0] repopath repotypes
if {"fossil" in $repotypes} {
#review - multiple process launches to fossil a bit slow on windows..
#review - multiple process launches to fossil a bit slow on windows..
#could we query global db in one go instead?
#
set fossil_prog [auto_execok fossil]
@ -444,14 +444,14 @@ namespace eval punk::mix::cli {
if {"project" in $repotypes} {
#punk project
if {![catch {package require textblock; package require patternpunk}]} {
set result [textblock::join -- [>punk . logo] " " $result]
set result [textblock::join -- [>punk . logo] " " $result]
append result \n
}
}
}
}
set timeline [exec fossil timeline -n 5 -t ci]
set timeline [string map {\r\n \n} $timeline]
append result $timeline
append result $timeline
if {$opt_v} {
set repostate [punk::repo::workingdir_state $repopath -repopaths $repopaths -repotypes fossil]
append result \n [punk::repo::workingdir_state_summary $repostate]
@ -516,7 +516,7 @@ namespace eval punk::mix::cli {
puts stderr "Use: >build_modules_from_source_to_base /x/src/modules2 /x/modules2 -subdirlist {skunkworks lib}"
exit 2
}
set srcdirname [file tail $srcdir]
set srcdirname [file tail $srcdir]
set build [file dirname $srcdir]/_build/$srcdirname ;#relative to *original* srcdir - not current_source_dir
if {[llength $subdirlist] == 0} {
@ -578,7 +578,7 @@ namespace eval punk::mix::cli {
}
set fileparts [split [file rootname $modpath] -]
#set tmfile_versionsegment [lindex $fileparts end]
lassign [split_modulename_version $modpath] basename tmfile_versionsegment
lassign [split_modulename_version $modpath] basename tmfile_versionsegment
if {$tmfile_versionsegment eq ""} {
#split_modulename_version version part will be empty if not valid tcl version
#last segment doesn't look even slightly versiony - fail.
@ -634,8 +634,8 @@ namespace eval punk::mix::cli {
set modulefile $buildfolder/$basename-$module_build_version.tm
$build_event targetset_init INSTALL $podtree_copy
$build_event targetset_addsource $current_source_dir/$modpath
$build_event targetset_init INSTALL $podtree_copy
$build_event targetset_addsource $current_source_dir/$modpath
if {$tmfile_versionsegment eq $magicversion} {
$build_event targetset_addsource $versionfile
}
@ -667,7 +667,7 @@ namespace eval punk::mix::cli {
if {[file exists $tmfile]} {
set newname $buildfolder/#modpod-$basename-$module_build_version/$basename-$module_build_version.tm
file rename $tmfile $newname
set tmfile $newname
set tmfile $newname
}
set fd [open $tmfile r]; fconfigure $fd -translation binary; set data [read $fd]; close $fd
set data [string map [list $magicversion $module_build_version] $data]
@ -745,12 +745,12 @@ namespace eval punk::mix::cli {
$build_event targetset_end SKIPPED
}
$build_event destroy
$build_installer destroy
$build_installer destroy
#JMN - review
#JMN - review
if {!$had_error} {
$event targetset_init INSTALL $target_module_dir/$basename-$module_build_version.tm
$event targetset_addsource $modulefile
$event targetset_init INSTALL $target_module_dir/$basename-$module_build_version.tm
$event targetset_addsource $modulefile
if {\
[llength [dict get [$event targetset_source_changes] changed]]\
|| [llength [$event get_targets_exist]] < [llength [$event get_targets]]\
@ -759,12 +759,12 @@ namespace eval punk::mix::cli {
$event targetset_started
# -- --- --- --- --- ---
if {$did_skip} {set did_skip 0; puts -nonewline stdout \n}
lappend module_list $modulefile
lappend module_list $modulefile
if {[catch {
file copy -force $modulefile $target_module_dir
} errMsg]} {
puts stderr "FAILED to copy zip modpod module $modulefile to $target_module_dir"
$event targetset_end FAILED -note "could not copy $modulefile"
$event targetset_end FAILED -note "could not copy $modulefile"
} else {
puts stderr "Copied zip modpod module $modulefile to $target_module_dir"
# -- --- --- --- --- ---
@ -782,7 +782,7 @@ namespace eval punk::mix::cli {
}
tarjar {
#basename may still contain #tarjar-
#to be obsoleted - update modpod to (optionally) use vfs::tar
#to be obsoleted - update modpod to (optionally) use vfs::tar
}
file {
set m $modpath
@ -808,12 +808,12 @@ namespace eval punk::mix::cli {
if {[file exists $current_source_dir/#tarjar-$basename-$magicversion]} {
#rebuild the .tm from the #tarjar
#rebuild the .tm from the #tarjar
if {[file exists $current_source_dir/#tarjar-$basename-$magicversion/DESCRIPTION.txt]} {
} else {
}
#REVIEW - should be in same structure/depth as $target_module_dir in _build?
@ -824,22 +824,22 @@ namespace eval punk::mix::cli {
set tmfile $buildfolder/$basename-$module_build_version.tm
file delete -force $buildfolder/#tarjar-$basename-$module_build_version
file delete -force $tmfile
file copy -force $current_source_dir/#tarjar-$basename-$magicversion $buildfolder/#tarjar-$basename-$module_build_version
#
#bsdtar doesn't seem to work.. or I haven't worked out the right options?
#exec tar -cvf $buildfolder/$basename-$module_build_version.tm $buildfolder/#tarjar-$basename-$module_build_version
package require tar
tar::create $tmfile $buildfolder/#tarjar-$basename-$module_build_version
tar::create $tmfile $buildfolder/#tarjar-$basename-$module_build_version
if {![file exists $tmfile]} {
puts stdout "ERROR: failed to build tarjar file $tmfile"
exit 4
}
#copy the file?
#set target $target_module_dir/$basename-$module_build_version.tm
#set target $target_module_dir/$basename-$module_build_version.tm
#file copy -force $tmfile $target
lappend module_list $tmfile
} else {
#assume that either the .tm is not a tarjar - or the tarjar dir is capped (trailing #) and the .tm has been manually tarred.
@ -851,7 +851,7 @@ namespace eval punk::mix::cli {
#
#set target_relpath [punkcheck::lib::path_relative $basedir $target_module_dir/$basename-$module_build_version.tm]
#set file_record [punkcheck::installfile_begin $basedir $target_relpath $installername -eventid $punkcheck_eventid]
$event targetset_init INSTALL $target_module_dir/$basename-$module_build_version.tm
$event targetset_init INSTALL $target_module_dir/$basename-$module_build_version.tm
$event targetset_addsource $versionfile
$event targetset_addsource $current_source_dir/$m
@ -902,7 +902,7 @@ namespace eval punk::mix::cli {
#------------------------------
}
continue
}
##------------------------------
@ -917,7 +917,7 @@ namespace eval punk::mix::cli {
#set changed_unchanged [punkcheck::recordlist::file_install_record_source_changes [lindex [dict get $file_record body] end]]
#set changed_list [dict get $changed_unchanged changed]
#----------
$event targetset_init INSTALL $target_module_dir/$m
$event targetset_init INSTALL $target_module_dir/$m
$event targetset_addsource $current_source_dir/$m
if {\
[llength [dict get [$event targetset_source_changes] changed]]\
@ -981,7 +981,7 @@ namespace eval punk::mix::cli {
}
if {$CALLDEPTH == 0} {
$event destroy
$installer destroy
$installer destroy
}
return $module_list
}
@ -1017,7 +1017,7 @@ namespace eval punk::mix::cli {
}
dict set kettle_reset_args $p $arglist
}
}
}
}
#call kettle_reinit to ensure recipes point to current project
@ -1095,14 +1095,14 @@ namespace eval punk::mix::cli {
kettle_reinit
}
}
set first [lindex $args 0]
set first [lindex $args 0]
if {[string match @* $first]} {
error "deck kettle doesn't support special operations - try calling tclsh kettle directly"
}
if {$first eq "-f"} {
set args [lassign $args __ path]
} else {
set path $startdir/build.tcl
set path $startdir/build.tcl
}
set opts [list]
@ -1123,9 +1123,9 @@ namespace eval punk::mix::cli {
}
}
#hardcoded kettle option names (::kettle option names) - retrieved using kettle::option names
#hardcoded kettle option names (::kettle option names) - retrieved using kettle::option names
#This is done so we don't have to load kettle lib for shell call (both loading as module and running shell are annoyingly SLOW)
#REVIEW - needs to be updated to keep in sync with kettle.
#REVIEW - needs to be updated to keep in sync with kettle.
set knownopts [list\
--exec-prefix --bin-dir --lib-dir --prefix --man-dir --html-dir --markdown-dir --include-dir \
--ignore-glob --dry --verbose --machine --color --state --config --with-shell --log \
@ -1202,7 +1202,7 @@ namespace eval punk::mix::cli {
package require punk::mix::base
package require punk::overlay
if {[catch {
punk::overlay::custom_from_base [namespace current] ::punk::mix::base
punk::overlay::custom_from_base [namespace current] ::punk::mix::base
} errM]} {
puts stderr "punk::mix::cli load error: Failed to overlay punk::mix::base $errM"
error "punk::mix::cli error: $errM"
@ -1213,9 +1213,9 @@ namespace eval punk::mix::cli {
# ++ +++ +++ +++ +++ +++ +++ +++ +++ +++ +++
## Ready
## Ready
package provide punk::mix::cli [namespace eval punk::mix::cli {
variable version
set version 999999.0a1.0
set version 999999.0a1.0
}]
return

6
src/modules/punk/mix/templates-999999.0a1.0.tm

@ -47,7 +47,7 @@ namespace eval punk::mix::templates {
lappend decls [list punk.templates {path src/decktemplates/vendor/punk pathtype currentproject vendor punk allowupdates 0 repo "https://www.gitea1.intx.com.au/jn/punkshell" reposubdir "src/decktemplates/vendor/punk"}]
lappend decls [list punk.isbogus {provider punk::mix::templates something blah}] ;#some capability for which there is no handler to validate - therefore no warning will result.
#review - we should report unhandled caps somewhere, or provide a mechanism to detect/report.
#we don't want to warn at the time this provider is loaded - as handler may legitimately be loaded later.
#we don't want to warn at the time this provider is loaded - as handler may legitimately be loaded later.
return $decls
}
}
@ -86,9 +86,9 @@ namespace eval punk::mix::templates {
# ++ +++ +++ +++ +++ +++ +++ +++ +++ +++ +++
## Ready
## Ready
package provide punk::mix::templates [namespace eval punk::mix::templates {
variable version
set version 999999.0a1.0
set version 999999.0a1.0
}]
return

12
src/modules/punk/mix/util-999999.0a1.0.tm

@ -57,7 +57,7 @@ namespace eval punk::mix::util {
incr i
set last_opt $i
} else {
set last_opt [expr {$i - 1}]
set last_opt [expr {$i - 1}]
break
}
}
@ -73,7 +73,7 @@ namespace eval punk::mix::util {
#puts stderr "opts: $opts paths: $paths"
#let's proceed, but warn the user if an apparent option is in paths
#let's proceed, but warn the user if an apparent option is in paths
foreach opt [list -encoding -eofchar -translation] {
if {$opt in $paths} {
puts stderr "fcat WARNING: apparent option $opt found after file argument(s) (expected them before filenames). Passing to fileutil::cat anyway - but for at least some versions, these options may be ignored. commandline 'fcat $args'"
@ -142,7 +142,7 @@ namespace eval punk::mix::util {
}
#----------------------------------------
#namespace import ::punk::ns::nsimport_noclobber
#namespace import ::punk::ns::nsimport_noclobber
proc namespace_import_pattern_to_namespace_noclobber {pattern ns} {
set source_ns [namespace qualifiers $pattern]
@ -153,7 +153,7 @@ namespace eval punk::mix::util {
set nscaller [uplevel 1 {namespace current}]
set ns [punk::nsjoin $nscaller $ns]
}
set a_export_patterns [namespace eval $source_ns {namespace export}]
set a_export_patterns [namespace eval $source_ns {namespace export}]
set a_commands [info commands $pattern]
set a_tails [lmap v $a_commands {namespace tail $v}]
set a_exported_tails [list]
@ -359,9 +359,9 @@ namespace eval punk::mix::util {
# ++ +++ +++ +++ +++ +++ +++ +++ +++ +++ +++
## Ready
## Ready
package provide punk::mix::util [namespace eval punk::mix::util {
variable version
set version 999999.0a1.0
set version 999999.0a1.0
}]
return

176
src/modules/punk/nav/fs-999999.0a1.0.tm

@ -21,7 +21,7 @@
#[manpage_begin punkshell_module_punk::nav::fs 0 999999.0a1.0]
#[copyright "2024"]
#[titledesc {punk::nav::fs console filesystem navigation}] [comment {-- Name section and table of contents description --}]
#[moddesc {fs nav}] [comment {-- Description at end of page heading --}]
#[moddesc {fs nav}] [comment {-- Description at end of page heading --}]
#[require punk::nav::fs]
#[keywords module filesystem terminal]
#[description]
@ -117,9 +117,9 @@ tcl::namespace::eval punk::nav::fs {
#Both tcl's notion of pwd and VIRTUAL_CWD can be out of sync with the process CWD. This happens when in a VFS.
#We can also have VIRTUAL_CWD navigate to spaces that Tcl's cd can't - review
variable VIRTUAL_CWD ;#cwd that tracks pwd except when in zipfs locations that are not at or below a mountpoint
variable VIRTUAL_CWD ;#cwd that tracks pwd except when in zipfs locations that are not at or below a mountpoint
if {![interp issafe]} {
set VIRTUAL_CWD [pwd]
set VIRTUAL_CWD [pwd]
} else {
set VIRTUAL_CWD ""
}
@ -127,25 +127,25 @@ tcl::namespace::eval punk::nav::fs {
variable VIRTUAL_CWD
set cwd [pwd]
if {$cwd ne $VIRTUAL_CWD} {
puts stderr "pwd: $cwd"
puts stderr "pwd: $cwd"
}
return $::punk::nav::fs::VIRTUAL_CWD
}
#TODO - maintain per 'volume/server' CWD
#e.g cd and ./ to:
# d:
#TODO - maintain per 'volume/server' CWD
#e.g cd and ./ to:
# d:
# //zipfs:
# //server
# https://example.com
# should return to the last CWD for that volume/server
#VIRTUAL_CWD follows pwd when changed via cd
set stackrecord [commandstack::rename_command -renamer punk::nav::fs cd {args} {
if {![catch {
$COMMANDSTACKNEXT {*}$args
} errM]} {
set ::punk::nav::fs::VIRTUAL_CWD [pwd]
set ::punk::nav::fs::VIRTUAL_CWD [pwd]
} else {
error $errM
}
@ -153,7 +153,7 @@ tcl::namespace::eval punk::nav::fs {
#*** !doctools
#[subsection {Namespace punk::nav::fs}]
#[para] Core API functions for punk::nav::fs
#[para] Core API functions for punk::nav::fs
#[list_begin definitions]
@ -170,7 +170,7 @@ tcl::namespace::eval punk::nav::fs {
#This seems unfortunate - as a multithreaded set of test runs might otherwise have made some sense.. but perhaps for tests more serious isolation is a good idea.
#It also seems common to cd when loading certain packages e.g tls from starkit.
#While in most/normal cases the library will cd back to the remembered working directory after only a brief time - there seem to be many opportunities for issues
#if the repl is used to launch/run a number of things in the one process
#if the repl is used to launch/run a number of things in the one process
proc d/ {args} {
variable VIRTUAL_CWD
@ -205,7 +205,7 @@ tcl::namespace::eval punk::nav::fs {
}
set dircount [llength [dict get $matchinfo dirs]]
set filecount [llength [dict get $matchinfo files]]
set symlinkcount [llength [dict get $matchinfo links]] ;#doesn't include windows shelllinks (.lnk)
set symlinkcount [llength [dict get $matchinfo links]] ;#doesn't include windows shelllinks (.lnk)
#set location [file normalize [dict get $matchinfo location]]
set location [dict get $matchinfo location]
@ -217,7 +217,7 @@ tcl::namespace::eval punk::nav::fs {
set filesizes [lsearch -all -inline -not $filesizes na]
set filebytes [tcl::mathop::+ {*}$filesizes]
lappend result filebytes [punk::lib::format_number $filebytes]
}
}
if {[punk::nav::fs::system::codethread_is_running]} {
if {[llength [info commands ::punk::console::titleset]]} {
#if ansi is off - punk::console::titleset will try 'local' api method - which can fail
@ -252,18 +252,18 @@ tcl::namespace::eval punk::nav::fs {
set a1 [lindex $args 0]
switch -exact -- $a1 {
. - ./ {
tailcall punk::nav::fs::d/
tailcall punk::nav::fs::d/
}
.. - ../ {
if {$VIRTUAL_CWD eq "//zipfs:/" && ![string match //zipfs:/* [pwd]]} {
#exit back to last nonzipfs path that was in use
set VIRTUAL_CWD [pwd]
tailcall punk::nav::fs::d/
tailcall punk::nav::fs::d/
}
#we need to use normjoin to allow navigation to //server instead of just to //server/share (//server browsing unimplemented - review)
# [file join //server ..] would become /server/.. - use normjoin to get //server
# file dirname //server/share would stay as //server/share
#we need to use normjoin to allow navigation to //server instead of just to //server/share (//server browsing unimplemented - review)
# [file join //server ..] would become /server/.. - use normjoin to get //server
# file dirname //server/share would stay as //server/share
#set up1 [file dirname $VIRTUAL_CWD]
set up1 [punk::path::normjoin $VIRTUAL_CWD ..]
if {[string match //zipfs:/* $up1]} {
@ -277,7 +277,7 @@ tcl::namespace::eval punk::nav::fs {
cd $up1
#set VIRTUAL_CWD [file normalize $a1]
}
tailcall punk::nav::fs::d/
tailcall punk::nav::fs::d/
}
}
@ -318,13 +318,13 @@ tcl::namespace::eval punk::nav::fs {
}
}
if {[file type $target] eq "directory"} {
set VIRTUAL_CWD $target
set VIRTUAL_CWD $target
}
}
tailcall punk::nav::fs::d/
}
set curdir $VIRTUAL_CWD
} else {
} else {
set curdir [pwd]
}
@ -365,7 +365,7 @@ tcl::namespace::eval punk::nav::fs {
set location $path
set glob *
if {$searchspec_relative} {
set searchbase [pwd]
set searchbase [pwd]
} else {
set searchbase $path
}
@ -373,19 +373,19 @@ tcl::namespace::eval punk::nav::fs {
set location [file dirname $path]
set glob [file tail $path] ;#search for exact match file
if {$searchspec_relative} {
set searchbase [pwd]
set searchbase [pwd]
} else {
set searchbase [file dirname $path]
}
}
}
set matchinfo [dirfiles_dict -searchbase $searchbase -tailglob $glob -with_sizes {f d l} -with_times {f d l} $location]
set matchinfo [dirfiles_dict -searchbase $searchbase -tailglob $glob -with_sizes {f d l} -with_times {f d l} $location]
#puts stderr "=--->$matchinfo"
set location [file normalize [dict get $matchinfo location]]
if {[string match //xzipfs:/* $location] || $location ne $last_location} {
#REVIEW - zipfs test disabled with leading x
#REVIEW - zipfs test disabled with leading x
#emit previous result
if {[dict size $this_result]} {
dict set this_result filebytes [punk::lib::format_number [dict get $this_result filebytes]]
@ -398,7 +398,7 @@ tcl::namespace::eval punk::nav::fs {
set this_result [dict create]
set dircount 0
set filecount 0
}
}
incr dircount [llength [dict get $matchinfo dirs]]
incr filecount [llength [dict get $matchinfo files]]
@ -406,7 +406,7 @@ tcl::namespace::eval punk::nav::fs {
dict set this_result location $location
dict set this_result dircount $dircount
dict set this_result filecount $filecount
set filesizes [dict get $matchinfo filesizes]
if {[llength $filesizes]} {
set filesizes [lsearch -all -inline -not $filesizes na]
@ -468,7 +468,7 @@ tcl::namespace::eval punk::nav::fs {
}
set normpath [file normalize $path]
cd $normpath
set matchinfo [dirfiles_dict -searchbase $normpath -with_sizes {f d l} -with_times {f d l} $normpath]
set matchinfo [dirfiles_dict -searchbase $normpath -with_sizes {f d l} -with_times {f d l} $normpath]
set dircount [llength [dict get $matchinfo dirs]]
set filecount [llength [dict get $matchinfo files]]
set location [file normalize [dict get $matchinfo location]]
@ -479,7 +479,7 @@ tcl::namespace::eval punk::nav::fs {
set filesizes [lsearch -all -inline -not $filesizes na]
set filebytes [tcl::mathop::+ {*}$filesizes]
lappend result filebytes [punk::lib::format_number $filebytes]
}
}
set out [dirfiles_dict_as_lines -stripbase 1 $matchinfo]
#return $out\n[pwd]
@ -583,7 +583,7 @@ tcl::namespace::eval punk::nav::fs {
set ext [file extension $path]
set extlower [string tolower $ext]
if {$extlower in $tcl_extensions} {
set newargs [lrange $args 1 end] ;#todo - fix to allow script in position other than first
set newargs [lrange $args 1 end] ;#todo - fix to allow script in position other than first
set ::argv0 $path
set ::argc [llength $newargs]
set ::argv $newargs
@ -609,7 +609,7 @@ tcl::namespace::eval punk::nav::fs {
}
}
if {$tcl_indicator} {
set newargs [lrange $args 1 end] ;#todo - fix to allow script in position other than first.
set newargs [lrange $args 1 end] ;#todo - fix to allow script in position other than first.
set ::argv0 $path
set ::argc [llength $newargs]
set ::argv $newargs
@ -645,7 +645,7 @@ tcl::namespace::eval punk::nav::fs {
}
proc dirfiles {args} {
set argd [punk::args::get_by_id ::punk::nav::fs::dirfiles $args]
lassign [dict values $argd] leaders opts values_dict
lassign [dict values $argd] leaders opts values_dict
set opt_stripbase [dict get $opts -stripbase]
set opt_formatsizes [dict get $opts -formatsizes]
@ -663,11 +663,11 @@ tcl::namespace::eval punk::nav::fs {
#dirfiles_dict would handle simple cases of globs within paths anyway - but we need to explicitly set tailglob here in all branches so that next level doesn't need to do file vs dir checks to determine user intent.
#(dir-listing vs file-info when no glob-chars present is inherently ambiguous so we test file vs dir to make an assumption - more explicit control via -tailglob can be done manually with dirfiles_dict)
if {$relativepath} {
set searchbase [pwd]
set searchbase [pwd]
if {!$has_tailglobs} {
if {[file isdirectory [file join $searchbase $searchspec]]} {
set location [file join $searchbase $searchspec]
set tailglob *
set tailglob *
} else {
set location [file dirname [file join $searchbase $searchspec]]
set tailglob [file tail $searchspec] ;#use exact match as a glob - will retrieve size,attributes etc.
@ -700,29 +700,29 @@ tcl::namespace::eval punk::nav::fs {
return [dirfiles_dict_as_lines -stripbase $opt_stripbase -formatsizes $opt_formatsizes $contents]
}
#todo - package as punk::nav::fs
#todo - package as punk::nav::fs
#todo - in thread
#todo - streaming version
#glob patterns in path prior to final segment should already be resolved before using dirfiles_dict - as the underlying filesystem mechanisms can't do nested globbing themselves.
#dirfiles_dict will assume the path up to the final segment is literal even if globchars are included therein.
#final segment globs will be recognised only if -tailglob is passed as empty string
#if -tailglob not supplied and last segment has globchars - presume searchspec parendir is the container and last segment is globbing within that.
#if -tailglob not supplied and last segment has no globchars - presume searchspec is a container(directory) and use glob *
#if -tailglob not supplied and last segment has no globchars - presume searchspec is a container(directory) and use glob *
#caller should use parentdir as location and set tailglob to search-pattern or exact match if location is intended to match a file rather than a directory
#examples:
# somewhere/files = search is effectively somewhere/files/* (location somewhere/files glob is *)
# somewhere/files/* = (as above)
# -tailglob * somewhere/files = (as above)
# -tailglob * somewhere/files = (as above)
#
# -tailglob "" somewhere/files = search somewhere folder for exactly 'files' (location somewhere glob is files)
# -tailglob files somewhere = (as above)
#
# somewhere/f* = search somewhere folder for f* (location somewhere glob is f*)
# -tailglob f* somewhere = (as above)
#
# -tailglob f* somewhere = (as above)
#
# This somewhat clumsy API is so that simple searches can be made in a default sensible manner without requiring extra -tailglob argument for the common cases - with lack of trailing glob segment indicating a directory listing
# - but we need to distinguish somewhere/files as a search of that folder vs somewhere/files as a search for exactly 'files' within somewhere, hence the -tailglob option to fine-tune.
# - this also in theory allows file/directory names to contain glob chars - although this is probably unlikely and/or unwise and not likely to be usable on all platforms.
# - this also in theory allows file/directory names to contain glob chars - although this is probably unlikely and/or unwise and not likely to be usable on all platforms.
#
#if caller supplies a tailglob as empty string - presume the caller hasn't set location to parentdir - and that last element is the search pattern.
# -searchbase is always passed through - and is only used to construct a location path if a relative searchspec was supplied
@ -733,7 +733,7 @@ tcl::namespace::eval punk::nav::fs {
-searchbase -default ""
-tailglob -default "\uFFFF"
#with_sizes & with_times must accept 0|1|f|d|l where f = files d = dirs l = links (punk::du)
-with_sizes -default "\uFFFF" -type string
-with_sizes -default "\uFFFF" -type string
-with_times -default "\uFFFF" -type string
@values -min 0 -max -1 -type string
}
@ -743,7 +743,7 @@ tcl::namespace::eval punk::nav::fs {
#puts stderr "searchspecs: $searchspecs [llength $searchspecs]"
#puts stdout "arglist: $opts"
if {[llength $searchspecs] > 1} {
#review - spaced paths ?
error "dirfiles_dict: multiple listing not *yet* supported"
@ -757,7 +757,7 @@ tcl::namespace::eval punk::nav::fs {
# -- --- --- --- --- --- ---
#we don't want to normalize..
#for example if the user supplies ../ we want to see ../result
#for example if the user supplies ../ we want to see ../result
set is_relativesearchspec [expr {[file pathtype $searchspec] eq "relative"}]
if {$opt_searchbase eq ""} {
@ -770,7 +770,7 @@ tcl::namespace::eval punk::nav::fs {
switch -- $opt_tailglob {
"" {
if {$searchspec eq ""} {
set location
set location
} else {
if {$is_relativesarchspec} {
#set location [file dirname [file join $opt_searchbase $searchspec]]
@ -821,13 +821,13 @@ tcl::namespace::eval punk::nav::fs {
set location $searchspec
}
}
set match_contents $opt_tailglob
set match_contents $opt_tailglob
}
}
#puts stdout "searchbase: $searchbase searchspec:$searchspec"
#file attr //cookit:/ returns {-vfs 1 -handle {}}
#file attr //cookit:/ returns {-vfs 1 -handle {}}
#we will treat it differently for now - use generic handler REVIEW
set in_vfs 0 ;#we use this only for a vfs which is reported to have a mountpoint by vfs::filesystem info - not all that have -vfs 1 attr like cookit.
if {[llength [package provide vfs]]} {
@ -873,11 +873,11 @@ tcl::namespace::eval punk::nav::fs {
#we don't really expect something like //c:/ , but anyway, it's not the same as c:/ and for all we know someone could use that as a volume name?
set in_other_pseudovol 1 ;#flag so we don't use twapi - hope generic can handle it (uses tcl glob)
} else {
#we could use 'file attr' here to test if {-vfs 1}
#but it's an extra filesystem hit on all normal paths too (which can be expensive on some systems)
#we could use 'file attr' here to test if {-vfs 1}
#but it's an extra filesystem hit on all normal paths too (which can be expensive on some systems)
#instead for now we'll assume any reasonable vfs should have been found by vfs::filesystem::info or mounted as a pseudovolume
}
}
}
@ -885,7 +885,7 @@ tcl::namespace::eval punk::nav::fs {
#relative vs absolute? review - cwd valid for //zipfs:/ ??
set listing [punk::du::lib::du_dirlisting_zipfs $location -glob $match_contents {*}$next_opt_with_sizes {*}$next_opt_with_times]
} elseif {$in_cookit} {
#seems to be a vfs - except it mounts on a pseudo-volume path //cookit:/
#seems to be a vfs - except it mounts on a pseudo-volume path //cookit:/
#don't use twapi
#could possibly use du_dirlisting_tclvfs REVIEW
#files and folders are all returned with the -types hidden option for glob on windows
@ -928,12 +928,12 @@ tcl::namespace::eval punk::nav::fs {
lappend dirs $vfsmount
}
}
}
}
#NOTE: -types {hidden d} * may return . & .. on unix platforms - but will not show them on windows.
#A mounted vfs exe (e.g sometclkit.exe) may be returned by -types {hidden d} on windows - but at the same time has "-hidden 0" in the result of file attr.
#non-unix platforms may have attributes to indicate hidden status even if filename doesn't have leading dot.
#mac & windows have these
#windows doesn't consider dotfiles as hidden - mac does (?)
@ -946,8 +946,8 @@ tcl::namespace::eval punk::nav::fs {
set flaggedhidden [punk::lib::lunique_unordered $flaggedhidden]
}
set dirs [lsort $dirs] ;#todo - natsort
set dirs [lsort $dirs] ;#todo - natsort
#foreach d $dirs {
@ -958,7 +958,7 @@ tcl::namespace::eval punk::nav::fs {
#glob -types {hidden} will not always return the combination of glob -types {hidden f} && -types {hidden d} (on windows anyway)
# -- ---
# -- ---
#can't lsort files without lsorting filesizes
#Note - the sort by index would convert an empty filesizes list to a list of empty strings - one for each entry in files
#We want to preserve the empty list if that's what the dirlisting mechanism returned (presumably because -with_sizes was 0 or explicitly excluded files)
@ -971,22 +971,22 @@ tcl::namespace::eval punk::nav::fs {
set sorted_filesizes [list]
foreach i $sortorder {
lappend sorted_files [lindex $files $i]
lappend sorted_filesizes [lindex $filesizes $i]
lappend sorted_filesizes [lindex $filesizes $i]
}
}
set files $sorted_files
set filesizes $sorted_filesizes
# -- ---
# -- ---
#jmn
foreach nm [list {*}$dirs {*}$files] {
if {[punk::winpath::illegalname_test $nm]} {
lappend nonportable $nm
}
}
}
set front_of_dict [dict create location $location searchbase $opt_searchbase]
set front_of_dict [dict create location $location searchbase $opt_searchbase]
set listing [dict merge $front_of_dict $listing]
set updated [dict create dirs $dirs files $files filesizes $filesizes nonportable $nonportable flaggedhidden $flaggedhidden underlayfiles $underlayfiles underlayfilesizes $underlayfilesizes]
@ -1045,7 +1045,7 @@ tcl::namespace::eval punk::nav::fs {
set prefix_test_list [tcl::prefix all $searchbases [lindex $shortest_to_longest 0 0]]
#if shortest doesn't match all searchbases - we have no common base
if {[llength $prefix_test_list] == [llength $searchbases]} {
set common_base [lindex $shortest_to_longest 0 0]; #we
set common_base [lindex $shortest_to_longest 0 0]; #we
}
}
}
@ -1082,11 +1082,11 @@ tcl::namespace::eval punk::nav::fs {
}
set $fileset $stripped
}
#Note: without fkeys we would need to remember to use common_base to rebuild (and file normalize!) the key when we need to query the dict-based elements: sizes & times - because we didn't strip those keys.
#Note: without fkeys we would need to remember to use common_base to rebuild (and file normalize!) the key when we need to query the dict-based elements: sizes & times - because we didn't strip those keys.
}
# -- --- --- --- --- --- --- --- --- --- ---
#assign symlinks to the dirs or files collection (the punk::du system doesn't sort this out
#assign symlinks to the dirs or files collection (the punk::du system doesn't sort this out
#As at 2024-09 for windows symlinks - Tcl can't do file readlink on symlinks created with mklink /D name target (SYMLINKD) or mklink name target (SYMLINK)
#We can't read the target information - best we can do is classify it as a file or a dir
#we can't use 'file type' as that will report just 'link' - but file isfile and file isdirectory work and should work for links on all platforms - REVIEW
@ -1110,7 +1110,7 @@ tcl::namespace::eval punk::nav::fs {
}
}
} else {
#fallback if no target_type
#fallback if no target_type
if {[file isfile $s]} {
lappend file_symlinks $s
#will be appended in finfo_plus later
@ -1125,9 +1125,9 @@ tcl::namespace::eval punk::nav::fs {
}
#we now have the issue that our symlinks aren't sorted within the dir/file categorisation - they currently will have to appear at beginning or end - TODO
# -- --- --- --- --- --- --- --- --- --- ---
#todo - sort whilst maintaining order for metadata?
#todo - sort whilst maintaining order for metadata?
#we need to co-sort files only with filesizes (other info such as times is keyed on fname so cosorting not required)
@ -1135,7 +1135,7 @@ tcl::namespace::eval punk::nav::fs {
if {$opt_formatsizes} {
set filesizes [punk::lib::format_number $filesizes] ;#accepts a list and will process each
}
#col2 (file info) with subcolumns
set widest2a [tcl::mathfunc::max {*}[lmap v [list {*}$files {*}$file_symlinks ""] {string length $v}]]
@ -1162,7 +1162,7 @@ tcl::namespace::eval punk::nav::fs {
set mtime [dict get $contents times $key m]
set ts [tcl::clock::format $mtime -format "%Y-%m-%d %H:%M:%S"]
} else {
#set ts [string repeat { } 19]
#set ts [string repeat { } 19]
set ts "$key vs [dict keys [dict get $contents times]]"
}
set note ""
@ -1181,7 +1181,7 @@ tcl::namespace::eval punk::nav::fs {
set mtime [dict get $contents times $key m]
set ts [tcl::clock::format $mtime -format "%Y-%m-%d %H:%M:%S"]
} else {
set ts "[string repeat { } 19]"
set ts "[string repeat { } 19]"
}
set note "link" ;#default only
if {[dict exists $contents linkinfo $key linktype]} {
@ -1207,24 +1207,24 @@ tcl::namespace::eval punk::nav::fs {
set fname [dict get $fdict file]
if {[file extension $fname] eq ".lnk"} {
if {![catch {package require punk::winlnk}]} {
set shortcutinfo [punk::winlnk::file_get_info $fname]
set shortcutinfo [punk::winlnk::file_get_info $fname]
set target_type "file" ;#default/fallback
if {[dict exists $shortcutinfo link_target]} {
set is_valid_lnk 1
set is_valid_lnk 1
set tgt [dict get $shortcutinfo link_target]
if {[file exists $tgt]} {
#file type could return 'link' - we will use isfile/isdirectory
if {[file isfile $tgt]} {
set target_type file
} elseif {[file isdirectory $tgt]} {
set target_type directory
set target_type directory
} else {
set target_type file ;## ?
}
} else {
#todo - see if punk::winlnk has info about the type at the time of linking
#for now - treat as file
}
}
} else {
#no link_target - probably an ordinary file - but there could have been some other error in reading the binary windows lnk format.
set is_valid_lnk 0
@ -1239,7 +1239,7 @@ tcl::namespace::eval punk::nav::fs {
}
directory {
#target of link is a dir - for display/categorisation purposes we want to see it as a dir
#will be styled later based on membership of dir_shortcuts
#will be styled later based on membership of dir_shortcuts
lappend dirs $fname
lappend dir_shortcuts $fname
}
@ -1292,7 +1292,7 @@ tcl::namespace::eval punk::nav::fs {
set fdisp ""
if {[string length $d]} {
if {$d in $flaggedhidden} {
set d1 [punk::ansi::a+ cyan normal]
set d1 [punk::ansi::a+ cyan normal]
}
if {$d in $vfsmounts} {
if {$d in $flaggedhidden} {
@ -1313,7 +1313,7 @@ tcl::namespace::eval punk::nav::fs {
}
} else {
if {$d in $nonportable} {
set d1 [punk::ansi::a+ red bold]
set d1 [punk::ansi::a+ red bold]
}
}
#dlink-style & dshortcut_style are for underlines - can be added with colours already set
@ -1336,11 +1336,11 @@ tcl::namespace::eval punk::nav::fs {
}
lappend displaylist [overtype::left $col1 $d1$d$RST]$f1$fdisp$RST
}
return [punk::lib::list_as_lines $displaylist]
}
}
#pass in base and platform to head towards purity/testability.
#pass in base and platform to head towards purity/testability.
#this function can probably never be pure in such a simple form - as it needs to read state from the os storage system configuration
#consider haskells approach of well-typed paths for cross-platform paths: https://hackage.haskell.org/package/path
#review: punk::winpath calls cygpath!
@ -1360,8 +1360,8 @@ tcl::namespace::eval punk::nav::fs {
set path_absolute [punk::unixywindows::towinpath $path]
#puts stderr "winpath: $path"
} else {
#todo handle volume-relative paths with volume specified c:etc c:
#note - tcl doesn't handle this properly anyway.. the win32 api should 'remember' the per-volume cwd
#todo handle volume-relative paths with volume specified c:etc c:
#note - tcl doesn't handle this properly anyway.. the win32 api should 'remember' the per-volume cwd
#not clear whether tcl can/will fix this - but it means these paths are dangerous.
#The cwd of the process can get out of sync with what tcl thinks is the working directory when you swap drives
#Arguably if ...?
@ -1421,14 +1421,14 @@ tcl::namespace::eval punk::nav::fs::lib {
tcl::namespace::path [tcl::namespace::parent]
#*** !doctools
#[subsection {Namespace punk::nav::fs::lib}]
#[para] Secondary functions that are part of the API
#[para] Secondary functions that are part of the API
#[list_begin definitions]
#proc utility1 {p1 args} {
# #*** !doctools
# #[call lib::[fun utility1] [arg p1] [opt {?option value...?}]]
# #[para]Description of utility1
# return 1
# #[para]Description of utility1
# return 1
#}
@ -1446,7 +1446,7 @@ tcl::namespace::eval punk::nav::fs::lib {
tcl::namespace::eval punk::nav::fs::system {
#*** !doctools
#[subsection {Namespace punk::nav::fs::system}]
#[para] Internal functions that are not part of the API
#[para] Internal functions that are not part of the API
#ordinary emission of chunklist when no repl
proc emit_chunklist {chunklist} {
@ -1471,18 +1471,18 @@ tcl::namespace::eval punk::nav::fs::system {
proc codethread_is_running {} {
if {[info commands ::punk::repl::codethread::is_running] ne ""} {
return [punk::repl::codethread::is_running]
return [punk::repl::codethread::is_running]
}
return 0
}
}
# ++ +++ +++ +++ +++ +++ +++ +++ +++ +++ +++
## Ready
## Ready
package provide punk::nav::fs [tcl::namespace::eval punk::nav::fs {
variable pkg punk::nav::fs
variable version
set version 999999.0a1.0
set version 999999.0a1.0
}]
return

24
src/modules/punk/repl-999999.0a1.0.tm

@ -127,7 +127,7 @@ namespace eval punk::repl {
puts stderr "\n*> repl background error: '$message'"
#puts stderr "*> [set ::errorInfo]"
puts stderr "*> errorinfo: [dict get $errdict -errorinfo]"
set stdinreader [fileevent stdin readable]
set stdinreader [chan event stdin readable]
if {![string length $stdinreader]} {
puts stderr "*> stdin reader inactive"
} else {
@ -420,14 +420,14 @@ proc repl::start {inchan args} {
puts stderr "-->repl::start active on $inchan $args replthread:[thread::id] codethread:$codethread"
set prompt_config [punk::repl::get_prompt_config]
doprompt "P% "
fileevent $inchan readable [list [namespace current]::repl_handler $inchan $prompt_config]
chan event $inchan readable [list [namespace current]::repl_handler $inchan $prompt_config]
set reading 1
#catch {
# set punk::console::tabwidth [punk::console::get_tabstop_apparent_width]
#}
vwait [namespace current]::done
fileevent $inchan readable {}
chan event $inchan readable {}
#puts stderr "-->start done = $::repl::done"
@ -1327,7 +1327,7 @@ proc repl::repl_handler {inputchan prompt_config} {
set prompt_reset_flag 0
}
fileevent $inputchan readable {}
chan event $inputchan readable {}
upvar ::punk::console::input_chunks_waiting input_chunks_waiting
#note -inputmode not available in Tcl 8.6 for chan configure!
#According to DKF - -buffering option doesn't affect input channels
@ -1542,14 +1542,14 @@ proc repl::repl_handler {inputchan prompt_config} {
#Re-enable channel read handler only if no waiting chunks - must process in order
##################################################################################
if {![llength $input_chunks_waiting($inputchan)]} {
fileevent $inputchan readable [list ::repl::repl_handler $inputchan $prompt_config]
chan event $inputchan readable [list ::repl::repl_handler $inputchan $prompt_config]
} else {
after idle [list ::repl::repl_handler $inputchan $prompt_config]
}
####################################################
} else {
#repl_handler_checkchannel $inputchan
fileevent $inputchan readable {}
chan event $inputchan readable {}
set reading 0
thread::send -async $::repl::codethread {set ::punk::repl::codethread::running 0}
if {$::tcl_interactive} {
@ -1757,7 +1757,7 @@ proc repl::repl_process_data {inputchan chunktype chunk stdinlines prompt_config
# #review
# rputs stderr "->0byte read stdin"
# if {[chan eof $inputchan]} {
# fileevent $inputchan readable {}
# chan event $inputchan readable {}
# set reading 0
# #set running 0
# if {$::tcl_interactive} {
@ -1973,7 +1973,7 @@ proc repl::repl_process_data {inputchan chunktype chunk stdinlines prompt_config
rputs stderr "-------------"
rputs stderr "$::errorInfo"
rputs stderr "-------------"
set stdinreader [fileevent $inputchan readable]
set stdinreader [chan event $inputchan readable]
if {![string length $stdinreader]} {
rputs stderr "*> $inputchan reader inactive"
} else {
@ -2185,7 +2185,7 @@ proc repl::repl_process_data {inputchan chunktype chunk stdinlines prompt_config
#chan configure stdout -buffering none
#JMN
fileevent $inputchan readable {}
chan event $inputchan readable {}
set reading 0
#don't let unknown use 'args' to convert commandstr to list
#===============================================================================
@ -2529,7 +2529,7 @@ proc repl::repl_process_data {inputchan chunktype chunk stdinlines prompt_config
#append commandstr \n
if {$::punk::repl::signal_control_c} {
set ::punk::repl::signal_control_c 0
fileevent $inputchan readable {}
chan event $inputchan readable {}
rputs stderr "* console_control: control-c"
flush stderr
set c [a yellow bold]
@ -2578,7 +2578,7 @@ proc repl::repl_process_data {inputchan chunktype chunk stdinlines prompt_config
}
#fileevent $inputchan readable [list repl::repl_handler $inputchan $prompt_config]
#chan event $inputchan readable [list repl::repl_handler $inputchan $prompt_config]
#catch {puts stderr "zend--->[rep $::arglej]"}
@ -2590,7 +2590,7 @@ proc repl::repl_process_data {inputchan chunktype chunk stdinlines prompt_config
rputs stderr "-------------"
rputs stderr "$::errorInfo"
rputs stderr "-------------"
set stdinreader [fileevent $inputchan readable]
set stdinreader [chan event $inputchan readable]
if {![string length $stdinreader]} {
rputs stderr "*> $inputchan reader inactive"
} else {

38
src/modules/punk/repl/codethread-999999.0a1.0.tm

@ -21,11 +21,11 @@
#[manpage_begin punkshell_module_punk::repl::codethread 0 999999.0a1.0]
#[copyright "2024"]
#[titledesc {Module repl codethread}] [comment {-- Name section and table of contents description --}]
#[moddesc {codethread for repl - root interpreter}] [comment {-- Description at end of page heading --}]
#[moddesc {codethread for repl - root interpreter}] [comment {-- Description at end of page heading --}]
#[require punk::repl::codethread]
#[keywords module repl]
#[description]
#[para] This is part of the infrastructure required for the punk::repl to operate
#[para] This is part of the infrastructure required for the punk::repl to operate
# ++ +++ +++ +++ +++ +++ +++ +++ +++ +++ +++
@ -122,7 +122,7 @@ tcl::namespace::eval punk::repl::codethread {
#*** !doctools
#[subsection {Namespace punk::repl::codethread}]
#[para] Core API functions for punk::repl::codethread
#[para] Core API functions for punk::repl::codethread
#[list_begin definitions]
@ -130,13 +130,13 @@ tcl::namespace::eval punk::repl::codethread {
#proc sample1 {p1 n args} {
# #*** !doctools
# #[call [fun sample1] [arg p1] [arg n] [opt {option value...}]]
# #[para]Description of sample1
# #[para]Description of sample1
# #[para] Arguments:
# # [list_begin arguments]
# # [arg_def tring p1] A description of string argument p1.
# # [arg_def integer n] A description of integer argument n.
# # [list_end]
# return "ok"
# return "ok"
#}
variable run_command_cache
@ -145,7 +145,7 @@ tcl::namespace::eval punk::repl::codethread {
#if {[catch {interp children}]} {
# #8.6.10 doesn't have it.. when was it introduced?
#} else {
#}
proc is_running {} {
@ -153,14 +153,14 @@ tcl::namespace::eval punk::repl::codethread {
return $running
}
proc runscript {script} {
#puts stderr "->runscript"
variable replthread_cond
variable replthread_cond
#variable output_stdout ""
#variable output_stderr ""
#expecting to be called from a thread::send in parent repl - ie in the toplevel interp so that the sub-interp "code" is available
#if a thread::send is done from the commandline in a codethread - Tcl will
#if a thread::send is done from the commandline in a codethread - Tcl will
if {![interp exists code] || ![info exists replthread_cond]} {
#in case someone tries calling from codethread directly - don't do anything or change any state
#(direct caller could create an interp named code at the level "" -> "code" -"code" and add a replthread_cond value to avoid this check - but it probably won't do anything useful)
@ -233,12 +233,12 @@ tcl::namespace::eval punk::repl::codethread {
flush stderr
#interp transfer code $errhandle ""
#flush $errhandle
#flush $errhandle
#set lastoutchar [string index [punk::ansi::ansistrip [interp eval code set ::punk::repl::codethread::output_stdout]] end]
#set lastoutchar [string index [punk::ansi::ansistrip [interp eval code [list set ::punk::repl::codethread::output_stdout]]] end]
set lastoutpart [interp eval code {string range $::punk::repl::codethread::output_stdout end-100 end}]
set lastoutpart [interp eval code {string range $::punk::repl::codethread::output_stdout end-100 end}]
#note we could be in a *large* ansi segment such as sixel data
#review - why do we need to ansistrip?
#review - why do we need to ansistrip?
set lastoutchar [string index [punk::ansi::ansistrip $lastoutpart] end]
#set lasterrchar [string index [punk::ansi::ansistrip [interp eval code set ::punk::repl::codethread::output_stderr]] end]
@ -247,7 +247,7 @@ tcl::namespace::eval punk::repl::codethread {
#puts stderr "-->[ansistring VIEW -lf 1 $lastoutchar$lasterrchar]"
set tid [thread::id]
tsv::set codethread_$tid info [list lastoutchar $lastoutchar lasterrchar $lasterrchar]
tsv::set codethread_$tid info [list lastoutchar $lastoutchar lasterrchar $lasterrchar]
tsv::set codethread_$tid status $status
tsv::set codethread_$tid result $result
tsv::set codethread_$tid errorcode $::errorCode
@ -277,14 +277,14 @@ tcl::namespace::eval punk::repl::codethread::lib {
tcl::namespace::path [tcl::namespace::parent]
#*** !doctools
#[subsection {Namespace punk::repl::codethread::lib}]
#[para] Secondary functions that are part of the API
#[para] Secondary functions that are part of the API
#[list_begin definitions]
#proc utility1 {p1 args} {
# #*** !doctools
# #[call lib::[fun utility1] [arg p1] [opt {?option value...?}]]
# #[para]Description of utility1
# return 1
# #[para]Description of utility1
# return 1
#}
@ -302,17 +302,17 @@ tcl::namespace::eval punk::repl::codethread::lib {
tcl::namespace::eval punk::repl::codethread::system {
#*** !doctools
#[subsection {Namespace punk::repl::codethread::system}]
#[para] Internal functions that are not part of the API
#[para] Internal functions that are not part of the API
}
# ++ +++ +++ +++ +++ +++ +++ +++ +++ +++ +++
## Ready
## Ready
package provide punk::repl::codethread [tcl::namespace::eval punk::repl::codethread {
variable pkg punk::repl::codethread
variable version
set version 999999.0a1.0
set version 999999.0a1.0
}]
return

56
src/modules/punk/sshrun-999999.0a1.0.tm

@ -8,7 +8,7 @@
# @@ Meta Begin
# Application punk::sshrun 999999.0a1.0
# Meta platform tcl
# Meta license ISC
# Meta license ISC
# @@ Meta End
# Copyright (c) 2009 Jose F. Nieves <nieves@ltp.uprrp.edu>
@ -33,14 +33,14 @@
#[manpage_begin punkshell_module_punk::sshrun 0 999999.0a1.0]
#[copyright "2009"]
#[titledesc {Tcl procedures to execute tcl scripts in remote hosts}] [comment {-- Name section and table of contents description --}]
#[moddesc {punk::sshrun tclssh clone}] [comment {-- Description at end of page heading --}]
#[moddesc {punk::sshrun tclssh clone}] [comment {-- Description at end of page heading --}]
#[require punk::sshrun]
#[keywords module ssh]
#[description]
#[para] This is a clone of tclssh by Jose F. Nieves
#[para] This is a clone of tclssh by Jose F. Nieves
#[para] The original repo is at: https://bitbucket.org/noaaport/tclssh/src/master/
#[para] This version is namespaced under punk::sshrun specifically for the Punk shell project - and may lag the original project or diverge.
#[para] You are encouraged to use the original Tclssh source from the above URL for your own projects
#[para] You are encouraged to use the original Tclssh source from the above URL for your own projects
# ++ +++ +++ +++ +++ +++ +++ +++ +++ +++ +++
@ -49,7 +49,7 @@
#[para] overview of punk::sshrun
#[para] SYNOPSIS
#[para] package require punk::sshrun
#[para] -
#[para] -
#[para] punk::sshrun::connect [lb]-t <tclsh_name>[rb] [lb]-- <ssh_options>[rb] [lb]<user>@[rb]<host>
#[para] Defaults: -t tclsh
#[subsection Concepts]
@ -127,22 +127,22 @@ namespace eval punk::sshrun {
#*** !doctools
#[subsection {Namespace punk::sshrun}]
#[para] Core API functions for punk::sshrun
#[para] Core API functions for punk::sshrun
#[list_begin definitions]
proc connect {args} {
#*** !doctools
#[call connect [arg args]]
#[para] Must be called first.
#[para] Must be called first.
#[para] This proc opens an io channel to the tclsh in the remote host (via ssh) that is kept in an internal variable for subsequent use.
#[para] The file handle can be retrieved if desired through the command: get_filehandle {host}
#[para] The file handle can be retrieved if desired through the command: get_filehandle {host}
variable ssh;
set usage {connect [-t <tclsh_name>] [-- <ssh_options>]
[<user>@]<host>};
set optlist {{t.arg "tclsh"}};
array set option [::cmdline::getoptions args $optlist $usage];
set cmd [concat "|ssh" $args $option(t) 2>@ stdout];
set F [open $cmd r+];
@ -200,7 +200,7 @@ namespace eval punk::sshrun {
# [call send [arg host]]
# [para]This proc does the equivalent of a
# [example {
# puts <filehandle> [join <script_list> \n]
# puts <filehandle> [join <script_list> \n]
# flush <filehandle>
# }]
variable ssh;
@ -242,9 +242,9 @@ namespace eval punk::sshrun {
# [example {
# [gets <filehandle> line]
# }]
upvar $line_varname line;
upvar $line_varname line;
variable ssh;
system::_verify_connection $host;
set r [gets $ssh($host,F) line];
return $r;
@ -264,9 +264,9 @@ namespace eval punk::sshrun {
# [para](see the send_exit proc above)
# [para]The function returns the number of lines read (0 if nothing is read before encoutering eof)
#
upvar $output_varname output;
upvar $output_varname output;
variable ssh;
system::_verify_connection $host;
set r 0;
@ -283,11 +283,11 @@ namespace eval punk::sshrun {
#*** !doctools
# [call pop_read [arg host] [arg numbytes] [arg output_varname]]
# [para] Returns: numbytes read. If numbytes is not positive, then read is called without the numbytes argument.
upvar $output_varname output;
upvar $output_varname output;
variable ssh;
system::_verify_connection $host;
if {$numbytes <= 0} {
set output [read $ssh($host,F)];
} else {
@ -306,7 +306,7 @@ namespace eval punk::sshrun {
# }]
variable ssh;
system::_verify_connection $host;
fileevent $ssh($host,F) $readable_writable $script;
chan event $ssh($host,F) $readable_writable $script;
}
proc hfconfigure {host args} {
@ -314,7 +314,7 @@ namespace eval punk::sshrun {
# [call hconfigure [arg host] [arg args]]
variable ssh;
system::_verify_connection $host;
eval fconfigure $ssh($host,F) $args;
eval chan configure $ssh($host,F) $args;
}
proc rexec {host script output_varname} {
@ -322,8 +322,8 @@ namespace eval punk::sshrun {
# [call rexec [arg host] [arg script] [arg output_varname]]
# [para] shortcut for:
# [example {
# ssh::rexec_nopop $host $script
# ssh::pop_all $host outputvar
# ssh::rexec_nopop $host $script
# ssh::pop_all $host outputvar
# }]
upvar $output_varname output;
rexec_nopop $host $script;
@ -392,7 +392,7 @@ namespace eval punk::sshrun {
# [call get_filehandle [arg host]]
variable ssh;
system::_verify_connection $host;
return $ssh($host,F);
}
@ -410,14 +410,14 @@ namespace eval punk::sshrun::lib {
namespace path [namespace parent]
#*** !doctools
#[subsection {Namespace punk::sshrun::lib}]
#[para] Secondary functions that are part of the API
#[para] Secondary functions that are part of the API
#[list_begin definitions]
#proc utility1 {p1 args} {
# #*** !doctools
# #[call lib::[fun utility1] [arg p1] [opt {?option value...?}]]
# #[para]Description of utility1
# return 1
# #[para]Description of utility1
# return 1
#}
@ -435,7 +435,7 @@ namespace eval punk::sshrun::lib {
namespace eval punk::sshrun::system {
#*** !doctools
#[subsection {Namespace punk::sshrun::system}]
#[para] Internal functions that are not part of the API
#[para] Internal functions that are not part of the API
#
# private
@ -452,11 +452,11 @@ namespace eval punk::sshrun::system {
}
# ++ +++ +++ +++ +++ +++ +++ +++ +++ +++ +++
## Ready
## Ready
package provide punk::sshrun [namespace eval punk::sshrun {
variable pkg punk::sshrun
variable version
set version 999999.0a1.0
set version 999999.0a1.0
}]
return

84
src/modules/punk/winrun-999999.0a1.0.tm

@ -37,7 +37,7 @@ namespace eval punk::winrun {
}
proc readchild_handler {chan hpid} {
#fileevent $chan readable {}
#chan event $chan readable {}
set data [read $chan 4096]
while {![chan blocked $chan] && ![eof $chan]} {
append data [read $chan 4096]
@ -46,19 +46,19 @@ namespace eval punk::winrun {
flush stdout
if {![eof $chan]} {
puts stdout "not eof $chan [fconfigure $chan] chan blocked:[chan blocked $chan]"
#fileevent $chan readable [list punk::winrun::readchild_handler $chan $hpid]
#chan event $chan readable [list punk::winrun::readchild_handler $chan $hpid]
} else {
#puts "eof: waiting exit process"
set punk::winrun::waitresult [twapi::wait_on_handle $hpid -wait -1]
}
}
proc readchilderr_handler {chan} {
fileevent $chan readable {}
chan event $chan readable {}
set data [read $chan]
puts stderr "err: $data"
flush stderr
if {![eof $chan]} {
fileevent $chan readable [list punk::winrun::readchild_handler $chan]
chan event $chan readable [list punk::winrun::readchild_handler $chan]
}
}
@ -81,13 +81,13 @@ namespace eval punk::winrun {
#after 1000
chan configure $readout -blocking 0
fileevent $readout readable [list readchild_handler $readout $hpid]
chan event $readout readable [list readchild_handler $readout $hpid]
puts stdout "input: [chan configure $writein]"
puts $writein "puts stdout blah;"
flush $writein
puts $writein "flush stdout"
flush $writein
puts $writein "puts exiting"
puts $writein "puts exiting"
puts $writein "after 10;exit 4"
flush $writein
#puts stdout x--[read $readout]
@ -106,13 +106,13 @@ namespace eval punk::winrun {
if {$waitresult eq "timeout"} {
puts stderr "tw_run: timeout waiting for process"
}
fileevent $readout readable {}
fileevent $readerr readable {}
chan event $readout readable {}
chan event $readerr readable {}
set code [twapi::get_process_exit_code $hpid]
twapi::close_handle $htid
twapi::close_handle $hpid
return [dict create exitcode $code]
return [dict create exitcode $code]
}
proc wait_on {hpid} {
set punk::winrun::waitresult [twapi::wait_on_handle $hpid -wait -1]
@ -130,7 +130,7 @@ namespace eval punk::winrun {
set code [twapi::get_process_exit_code $hpid]
twapi::close_handle $htid
twapi::close_handle $hpid
return [dict create exitcode $code]
return [dict create exitcode $code]
}
#completely raw to windows createprocess API - caller will really need to understand what they're doing.
@ -205,10 +205,10 @@ namespace eval punk::winrun {
append cmdline {"}
set chars [split $w ""]
set wordlen [string length $w]
set nlast [expr {$wordlen -1}]
set nlast [expr {$wordlen -1}]
for {set n 0} {$n<$wordlen} {incr n} {
set char [lindex $chars $n]
set num_backslashes 0
set num_backslashes 0
while {$char eq "\\" && $n<$nlast} {
incr num_backslashes
incr n
@ -216,7 +216,7 @@ namespace eval punk::winrun {
}
if {$n > $nlast} {
append cmdline [string repeat "\\" [expr {$num_backslashes * 2}]]
break
break
} elseif {$char eq {"}} {
#escape all backslashes and the following double-quote
append cmdline [string repeat "\\" [expr {$num_backslashes * 2 + 1}]] $char
@ -234,7 +234,7 @@ namespace eval punk::winrun {
puts stdout --cmdline->$cmdline
}
# -----------------
#tw_run $cmdline
#tw_run $cmdline
#assertion - can be treated as tcl list ?
return $cmdline
}
@ -333,8 +333,8 @@ namespace eval punk::winrun {
if {[lindex $chars $n+1] eq {"}} {
incr n ;#move to second {"}
} else {
set copychar false
set in_doublequote_part 0
set copychar false
set in_doublequote_part 0
}
} else {
set copychar false
@ -350,7 +350,7 @@ namespace eval punk::winrun {
break
}
if {$copychar} {
append p [lindex $chars $n]
append p [lindex $chars $n]
}
}
set rem [string range $cmdline $n+1 end]
@ -362,7 +362,7 @@ namespace eval punk::winrun {
tw_run [quote_win {*}$args]
}
#an experiment - this is essentially an identity transform unless flags are set. - result afer cmd.exe processes escapes is the same as running raw with no quoting
#an experiment - this is essentially an identity transform unless flags are set. - result afer cmd.exe processes escapes is the same as running raw with no quoting
#this follows the advice of 'don't let cmd see any double quotes unescaped' - but that's effectively a pretty useless strategy.
#The -useprequoted and -usepreescaped flags are the only difference
#these rely on the fact we can prepend a caret to each argument without affecting the resulting string - and use that as an indicator to treat specific input 'arguments' differently i.e by keeping existing escapes only.
@ -385,7 +385,7 @@ namespace eval punk::winrun {
set cmdline ""
set i 0
set meta_chars [list {"} "(" ")" ^ < > & |]
set meta_chars [list {"} "(" ")" ^ < > & |]
#note that %var% and !var! work the same whether within a double quote section or not
if {$disallowvars} {
lappend meta_chars % !
@ -398,8 +398,8 @@ namespace eval punk::winrun {
foreach w $tcl_list {
set qword ""
set wordlen [string length $w]
set nlast [expr {$wordlen -1}]
set chars [split $w ""]
set nlast [expr {$wordlen -1}]
set chars [split $w ""]
set wordlen [string length $w]
set nlast [expr {$wordlen -1}]
@ -514,14 +514,14 @@ namespace eval punk::winrun {
#??
}
#if %var% was in original string - a variable named %"var"% can be substituted after we have done our quoting.
#no matter what quoting scheme we use - there will be a corresponding string between %'s that can in theory be exploited if
#no matter what quoting scheme we use - there will be a corresponding string between %'s that can in theory be exploited if
if {$in_quotes} {
#note that the *intended* quoting will be opposite to the resultant quoting from wrapping with quote_win
#therefore, counterintuitively we can enable the var when in_quotes is true here, and &cmd won't run.
#double quotes in the var don't seem to cause cmd.exe to change it's concept of in_quotes so &cmd also won't run
#However.. backspace can can break quoting. e.g \b&cmd
#However.. backspace can can break quoting. e.g \b&cmd
if {$allowvars} {
append qword [lindex $chars $n]
append qword [lindex $chars $n]
} else {
append qword {"} [lindex $chars $n] {"} ;#add in pairs so we don't disturb structure for argv
}
@ -544,7 +544,7 @@ namespace eval punk::winrun {
if {$in_quotes} {
append qword {"^^"} ;#add quotes in pairs so we don't disturb structure for argv
} else {
append qword {^^}
append qword {^^}
}
} else {
if {[lindex $chars $n] in $meta_chars} {
@ -559,7 +559,7 @@ namespace eval punk::winrun {
}
}
append cmdline $qword " "
}
set cmdline [string range $cmdline 0 end-1]
if {$verbose} {
@ -567,32 +567,32 @@ namespace eval punk::winrun {
}
return $cmdline
}
# - This approach with repeated double quotes gives inconsistent behaviour between twapi CommandLineToArgvW and tclsh -
# - This approach with repeated double quotes gives inconsistent behaviour between twapi CommandLineToArgvW and tclsh -
#prepare arguments that are given to cmd.exe such that they will be passed through to an executable that uses standard windows commandline parsing such as CommandLineToArgvW
#for each arg:
#double up any backslashes that precede double quotes, double up existing double quotes - then wrap in a single set of double quotes if argument had any quotes in it.
#This doesn't use \" or ^ style escaping - but the 2008+ argv processing on windows supposedly does what we want with doubled-up quotes and slashes, and cmd.exe passes them through
#In practice - it seems less consistent/reliable
#In practice - it seems less consistent/reliable
proc quote_cmdpassthru_test {args} {
lassign [internal::get_run_opts $args] _r runopts _c cmdargs
set allowvars [expr {"-allowvars" in $runopts}]
set verbose [expr {"-verbose" in $runopts}]
set tcl_list [lmap v $cmdargs {internal::objclone $v}]
set meta_chars [list {"} "(" ")" ^ < > & |]
set meta_chars [list {"} "(" ")" ^ < > & |]
if {!$allowvars} {
lappend meta_chars % !
}
set cmdline ""
foreach w $tcl_list {
set chars [split $w ""]
set chars [split $w ""]
set wordlen [llength $chars]
#set nlast [expr {$wordlen -1}]
set qword ""
for {set n 0} {$n<$wordlen} {incr n} {
set num_slashes 0
while {[lindex $chars $n] eq "\\" && $n<$wordlen} {
incr num_slashes
incr num_slashes
incr n
}
if {[lindex $chars $n] eq {"}} {
@ -615,7 +615,7 @@ namespace eval punk::winrun {
return $cmdline
}
#caret quoting of all meta_chars
#caret quoting of all meta_chars
proc quote_cmdblock {args} {
lassign [internal::get_run_opts $args] _r runopts _c cmdargs
set allowvars [expr {"-allowvars" in $runopts}]
@ -624,7 +624,7 @@ namespace eval punk::winrun {
set tcl_list [lmap v $cmdargs {internal::objclone $v}]
set cmdline ""
set i 0
set meta_chars [list "(" ")" ^ < > & |]
set meta_chars [list "(" ")" ^ < > & |]
if {!$allowvars} {
lappend meta_chars % !
}
@ -633,8 +633,8 @@ namespace eval punk::winrun {
}
foreach w $tcl_list {
set wordlen [string length $w]
set nlast [expr {$wordlen -1}]
set chars [split $w ""]
set nlast [expr {$wordlen -1}]
set chars [split $w ""]
foreach char $chars {
if {$char in $meta_chars} {
append cmdline "^$char"
@ -663,8 +663,8 @@ namespace eval punk::winrun {
set cmd_in_quotes 0
foreach w $tcl_list {
set wordlen [string length $w]
set nlast [expr {$wordlen -1}]
set chars [split $w ""]
set nlast [expr {$wordlen -1}]
set chars [split $w ""]
foreach char $chars {
if {$char eq {"}} {
append cmdline {^"}
@ -704,7 +704,7 @@ namespace eval punk::winrun {
#round-trip test
#use standard(!) win arg quoting first - then deconstruct using the win32 api, and the tcl implementation
#use standard(!) win arg quoting first - then deconstruct using the win32 api, and the tcl implementation
proc testrawline {rawcmdline} {
puts "input string : $rawcmdline"
set win_argv [unquote_win $rawcmdline]
@ -770,7 +770,7 @@ namespace eval punk::winrun {
#get_run_opts - allow completely arbitrary commandline following controlling flags - with no collision issues if end-of-opts flag "--" is used.
#singleton flags allowed preceding commandline. (no support for option-value pairs in the controlling flags)
#This precludes use of executable beginning with a dash unless -- provided as first argument or with only known run-opts preceding it.
#This precludes use of executable beginning with a dash unless -- provided as first argument or with only known run-opts preceding it.
#This should allow any first word for commandlist even -- itself if a protective -- provided at end of any arguments intended to control the function.
proc get_run_opts {arglist} {
if {[catch {
@ -852,7 +852,7 @@ namespace eval punk::winrun {
set nscaller [uplevel 1 {namespace current}]
set ns [punk::nsjoin $nscaller $ns]
}
set a_export_patterns [namespace eval $source_ns {namespace export}]
set a_export_patterns [namespace eval $source_ns {namespace export}]
set a_commands [info commands $pattern]
set a_tails [lmap v $a_commands {namespace tail $v}]
set a_exported_tails [list]
@ -893,9 +893,9 @@ namespace eval punk::winrun {
# ++ +++ +++ +++ +++ +++ +++ +++ +++ +++ +++
## Ready
## Ready
package provide punk::winrun [namespace eval punk::winrun {
variable version
set version 999999.0a1.0
set version 999999.0a1.0
}]
return

128
src/modules/punkcheck-0.1.0.tm

@ -69,7 +69,7 @@ namespace eval punkcheck {
}
proc load_records_from_file {punkcheck_file} {
proc load_records_from_file {punkcheck_file} {
set record_list [list]
if {[file exists $punkcheck_file]} {
set tdlscript [punk::mix::util::fcat $punkcheck_file]
@ -86,7 +86,7 @@ namespace eval punkcheck {
set linecount [llength [split $newtdl \n]]
#puts stdout $newtdl
set fd [open $punkcheck_file w]
fconfigure $fd -translation binary
chan configure $fd -translation binary
puts -nonewline $fd $newtdl
close $fd
return [list recordcount [llength $recordlist] linecount $linecount]
@ -94,7 +94,7 @@ namespace eval punkcheck {
#todo - work out way to use same punkcheck file for multiple installers running concurrently. Thread?
#an installtrack objects represents an installation path from sourceroot to targetroot
#an installtrack objects represents an installation path from sourceroot to targetroot
#The source and target folders should be as specific as possible but it is valid to specify for example c:/ -> c:/ (or / -> /) if source and targets within the installation operation are spread around.
#
set objname [namespace current]::installtrack
@ -104,7 +104,7 @@ namespace eval punkcheck {
#FILEINFO record - target fileset with body records: INSTALL-RECORD,INSTALL-INPROGRESS,INSTALL-SKIPPED,DELETE-RECORD,DELETE-INPROGRESS,MODIFY-INPROGRESS,MODIFY-RECORD
#each FILEINFO body being a list of SOURCE records
oo::class create targetset {
variable o_targets
variable o_targets
variable o_keep_installrecords
variable o_keep_skipped
variable o_keep_inprogress
@ -132,7 +132,7 @@ namespace eval punkcheck {
-keep_inprogress $o_keep_inprogress\
body $o_records
}
#retrieve last completed record for the fileset ie exclude SKIPPED,INSTALL-INPROGRESS,DELETE-INPROGRESS,MODIFY-INPROGRESS
method get_last_record {fileset_record} {
set body [dict_getwithdefault $fileset_record body [list]]
@ -189,11 +189,11 @@ namespace eval punkcheck {
}
set o_ts_end [dict get $opts -tsend]
set o_types [dict get $opts -types]
set o_configdict [dict get $opts -config]
set o_configdict [dict get $opts -config]
set o_rel_sourceroot $rel_sourceroot
set o_rel_targetroot $rel_targetroot
}
}
destructor {
#puts "[self] destructor called"
}
@ -339,14 +339,14 @@ namespace eval punkcheck {
set installing_record [lindex $fileinfo_body end]
set ts_start [dict get $installing_record -ts]
set ts_now [clock microseconds]
set ts_now [clock microseconds]
set metadata_us [expr {$ts_now - $ts_start}]
dict set installing_record -metadata_us $metadata_us
dict set installing_record -ts_start_transfer $ts_now
lset fileinfo_body end $installing_record
return [dict set o_fileset_record body $fileinfo_body]
} else {
#legacy call
@ -368,7 +368,7 @@ namespace eval punkcheck {
}
set status [string toupper $status]
set statusdict [dict create OK RECORD SKIPPED SKIPPED FAILED FAILED]
set statusdict [dict create OK RECORD SKIPPED SKIPPED FAILED FAILED]
if {$o_operation_start_ts eq ""} {
error "[self] targetset_end $status - no current operation - call targetset_started first"
}
@ -383,7 +383,7 @@ namespace eval punkcheck {
error "targetset_end $status error. targetlist mismatch between file : $targetlist vs $o_targets"
}
set operation_end_ts [clock microseconds]
set elapsed_us [expr {$operation_end_ts - $o_operation_start_ts}]
set elapsed_us [expr {$operation_end_ts - $o_operation_start_ts}]
set file_record_body [dict get $o_fileset_record body]
set installing_record [lindex $file_record_body end]
set punkcheck_file [$o_installer get_checkfile]
@ -414,12 +414,12 @@ namespace eval punkcheck {
}
}
set cksum_us [expr {[clock microseconds] - $ts_begin_cksum}]
dict set installing_record -targets_cksums $new_targets_cksums
dict set installing_record -targets_cksums $new_targets_cksums
dict set installing_record -cksum_all_opts $cksum_all_opts
dict set installing_record -cksum_us $cksum_us
}
lset file_record_body end $installing_record
dict set o_fileset_record body $file_record_body
dict set o_fileset_record body $file_record_body
set o_fileset_record [punkcheck::recordlist::file_record_prune $o_fileset_record]
set oldrecordinfo [punkcheck::recordlist::get_file_record $targetlist $record_list]
@ -436,8 +436,8 @@ namespace eval punkcheck {
set o_operation ""
return $o_fileset_record
}
#can supply empty cksum value
# - that will influence the opts used if there is no existing install record
#can supply empty cksum value
# - that will influence the opts used if there is no existing install record
method targetset_cksumcache_set {path_cksum_dict} {
set o_path_cksum_cache $path_cksum_dict
}
@ -504,12 +504,12 @@ namespace eval punkcheck {
variable o_ts
variable o_keep_events
variable o_checkfile
variable o_sourceroot
variable o_sourceroot
variable o_rel_sourceroot
variable o_targetroot
variable o_rel_targetroot
variable o_record_list
variable o_active_event
variable o_active_event
variable o_events
constructor {installername punkcheck_file} {
set o_active_event ""
@ -546,7 +546,7 @@ namespace eval punkcheck {
#$o_events add $e [dict get $e -id]
$o_events add $eobj [dict get $e -id]
}
}
destructor {
#puts "[self] destructor called"
@ -562,7 +562,7 @@ namespace eval punkcheck {
}
#call set_source_target before calling start_event/end_event
#each event can have different source->target pairs - but may often have same, so set on installtrack as defaults. Only persisted in event records.
#each event can have different source->target pairs - but may often have same, so set on installtrack as defaults. Only persisted in event records.
method set_source_target {sourceroot targetroot} {
if {[file pathtype $sourceroot] ne "absolute"} {
error "[self] set_source_target error: sourceroot must be absolute path. Received '$sourceroot'"
@ -605,7 +605,7 @@ namespace eval punkcheck {
}
method save_installer_record {} {
set file_records [punkcheck::load_records_from_file $o_checkfile]
set this_installer_record [my as_record]
set persistedinfo [punkcheck::recordlist::get_installer_record $o_name $file_records]
@ -658,13 +658,13 @@ namespace eval punkcheck {
set resultinfo [punkcheck::recordlist::get_installer_record $o_name $o_record_list]
}
method get_recordlist {} {
return $o_recordlist
return $o_recordlist
}
method end_event {} {
if {$o_active_event eq ""} {
error "[self] end_event error - no active event"
}
$o_active_event end
$o_active_event end
}
method get_event {} {
return $o_active_event
@ -720,7 +720,7 @@ namespace eval punkcheck {
append msg "Call in order:" \n
append msg " start_installer_event (get dict with eventid and recordset keys)"
append msg " installfile_begin (to return a new INSTALLING record) - must pass in a valid eventid" \n
append msg " installfile_add_source_and_fetch_metadata (1+ times to update SOURCE record with checksum/timestamp info from source)" \n
append msg " installfile_add_source_and_fetch_metadata (1+ times to update SOURCE record with checksum/timestamp info from source)" \n
append msg " ( - possibly with same algorithm as previous installrecord)" \n
append msg " ( - todo - search/load metadata for this source from other FILEINFO records for same installer)" \n
append msg "Finalize by calling:" \n
@ -749,7 +749,7 @@ namespace eval punkcheck {
set punkcheck_file [file join $punkcheck_folder/.punkcheck]
set record_list [load_records_from_file $punkcheck_file]
set resultinfo [punkcheck::recordlist::get_installer_record $installername $record_list]
set installer_record_position [dict get $resultinfo position]
if {$installer_record_position == -1} {
@ -805,7 +805,7 @@ namespace eval punkcheck {
#validate any passed cached_cksums
foreach cacheinfo $cached_cksums {
if {[llength $cacheinfo] % 2 != 0} {
error "installfile_add_source_and_fetch_metadata error.If cached_cksums is supplied, it must be a list of dicts containing keys cksum & opts"
error "installfile_add_source_and_fetch_metadata error.If cached_cksums is supplied, it must be a list of dicts containing keys cksum & opts"
}
dict for {k v} $cacheinfo {
switch -- $k {
@ -814,7 +814,7 @@ namespace eval punkcheck {
#todo - validate $v keys
}
default {
error "installfile_add_source_and_fetch_metadata error. Unrecognised key $k. Known keys {cksum opts}"
error "installfile_add_source_and_fetch_metadata error. Unrecognised key $k. Known keys {cksum opts}"
}
}
@ -837,7 +837,7 @@ namespace eval punkcheck {
}
}
}
#check that this relpath not already added as child of *-INPROGRESS
#check that this relpath not already added as child of *-INPROGRESS
set file_record_body [dict_getwithdefault $file_record body [list]] ;#new file_record may have no body
set installing_record [lindex $file_record_body end]
set already_present_record [lib::install_record_get_matching_source_record $installing_record $source_relpath]
@ -871,14 +871,14 @@ namespace eval punkcheck {
#use first entry in cached_cksums if we can
if {[llength $cached_cksums]} {
set use_cache 1
set use_cache_record [lindex $cached_cksums 0]
set use_cache_record [lindex $cached_cksums 0]
}
}
#todo - accept argument of cached source cksum info (for client calling multiple targets with same source in quick succession e.g when building .vfs kits with multiple runtimes)
#if same cksum_opts - then use cached data instead of checksumming here.
#allow nonexistant as a source
#allow nonexistant as a source
set fpath [file join $punkcheck_folder $source_relpath]
if {![file exists $fpath]} {
set ftype "missing"
@ -939,14 +939,14 @@ namespace eval punkcheck {
set installing_record [lindex $file_record_body end]
set ts_start [dict get $installing_record -ts]
set ts_now [clock microseconds]
set ts_now [clock microseconds]
set metadata_us [expr {$ts_now - $ts_start}]
dict set installing_record -metadata_us $metadata_us
dict set installing_record -ts_start_transfer $ts_now
lset file_record_body end $installing_record
dict set file_record body $file_record_body
@ -983,7 +983,7 @@ namespace eval punkcheck {
dict set installing_record tag "INSTALL-RECORD"
lset file_record_body end $installing_record
dict set file_record body $file_record_body
dict set file_record body $file_record_body
set file_record [punkcheck::recordlist::file_record_prune $file_record]
@ -1016,8 +1016,8 @@ namespace eval punkcheck {
set tsnow [clock microseconds]
set elapsed_us [expr {$tsnow - $ts_start}]
dict set installing_record -elapsed_us $elapsed_us
dict set installing_record tag "INSTALL-SKIPPED"
dict set installing_record tag "INSTALL-SKIPPED"
lset file_record_body end $installing_record
dict set file_record body $file_record_body
@ -1076,7 +1076,7 @@ namespace eval punkcheck {
#should work on *-INPROGRESS or INSTALL(etc) record - don't restrict tag to INSTALL
proc install_record_get_matching_source_record {install_record source_relpath} {
set body [dict_getwithdefault $install_record body [list]]
set body [dict_getwithdefault $install_record body [list]]
foreach src $body {
if {[dict get $src tag] eq "SOURCE"} {
if {[dict_getwithdefault $src -path ""] eq $source_relpath} {
@ -1124,7 +1124,7 @@ namespace eval punkcheck {
set do_normalize 1
}
} else {
#case differences in volumes is common on windows
#case differences in volumes is common on windows
set do_normalize 1
}
if {$do_normalize} {
@ -1207,7 +1207,7 @@ namespace eval punkcheck {
if {[dict exists $dictValue {*}$keys]} {
return [dict get $dictValue {*}$keys]
} else {
return [lindex $args end]
return [lindex $args end]
}
}
lappend PUNKARGS [list {
@ -1273,11 +1273,11 @@ namespace eval punkcheck {
# -overwrite newer-targets will copy files with older source timestamp over newer target timestamp and those missing at the target (a form of 'restore' operation)
# -overwrite older-targets will copy files with newer source timestamp over older target timestamp and those missing at the target
# -overwrite all-targets will copy regardless of timestamp at target
# -overwrite installedsourcechanged-targets will copy if the target doesn't exist or the source changed
# -overwrite installedsourcechanged-targets will copy if the target doesn't exist or the source changed
# -overwrite synced-targets will copy if the target doesn't exist or the source changed and the target cksum is the same as the last INSTALL-RECORD -targets_cksums entry
# review - timestamps unreliable
# - what about slightly mismatched system clocks and mounted filesystems? caller responsibility to verify first?
# if timestamp exactly equal - should we check content-hash? This is presumably only likely to occur deliberately(maliciously?)
# - what about slightly mismatched system clocks and mounted filesystems? caller responsibility to verify first?
# if timestamp exactly equal - should we check content-hash? This is presumably only likely to occur deliberately(maliciously?)
# e.g some process that digitally signs or otherwise modifies a file and preserves update timestmp?
# if such a content-mismatch - what default behaviour and what options would make sense?
# probably it's reasonable that only all-targets would overwrite such files.
@ -1369,7 +1369,7 @@ namespace eval punkcheck {
if {[llength [file split $af]] > 1} {
error "punkcheck::install received invalid -antiglob_file entry '$af'. -antiglob_file entries are meant to match to a file name at any level so cannot contain path separators"
}
}
}
# -- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- ---
set opt_antiglob_dir_core [dict get $opts -antiglob_dir_core]
if {$opt_antiglob_dir_core eq "\uFFFF"} {
@ -1383,7 +1383,7 @@ namespace eval punkcheck {
if {[llength [file split $ad]] > 1} {
error "punkcheck::install received invalid -antiglob_dir entry '$ad'. -antiglob_dir entries are meant to match to a directory name at any level so cannot contain path separators"
}
}
}
# -- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- ---
set opt_antiglob_paths [dict get $opts -antiglob_paths] ;#todo - combine with config file in source tree .punkcheckpublish (?)
#antiglob_paths will usually contain file separators - and may contain glob patterns within each segment
@ -1482,7 +1482,7 @@ namespace eval punkcheck {
} else {
set store_source_cksums 0
}
@ -1545,12 +1545,12 @@ namespace eval punkcheck {
}
if {$suppress == 0} {
lappend match_list $m
}
}
}
#sample .punkcheck file record (raw form) to make the code clearer
#punk::tdl converts to dict form e.g: tag FILEINFO -targets filename body sublist
#Valid installrecord types are INSTALL-RECORD SKIPPED INSTALL-INPROGRESS, MODIFY-RECORD MODIFY-INPROGRESS DELETE-RECORD DELETE-INPROGRESS
#Valid installrecord types are INSTALL-RECORD SKIPPED INSTALL-INPROGRESS, MODIFY-RECORD MODIFY-INPROGRESS DELETE-RECORD DELETE-INPROGRESS
#
#FILEINFO -targets jjjetc-0.1.0.tm -keep_installrecords 2 -keep_skipped 1 -keep_inprogress 2 {
# INSTALL-RECORD -tsiso 2023-09-20T07:30:30 -ts 1695159030266610 -installer punk::mix::cli::build_modules_from_source_to_base -metadata_us 18426 -ts_start_transfer 1695159030285036 -transfer_us 10194 -elapsed_us 28620 {
@ -1563,15 +1563,15 @@ namespace eval punkcheck {
# }
#}
if {[llength $match_list]} {
if {[llength $match_list]} {
#example - target dir has a file where there is a directory at the source
if {[file exists $current_target_dir] && ([file type $current_target_dir] ni [list directory])} {
error "punkcheck::install target subfolder $current_target_dir exists but is not of type 'directory'. Type current target folder: [file type $current_target_dir]"
}
}
#proc get_relativecksum_from_base_and_fullpath {base fullpath args}
#puts stdout "Current target dir: $current_target_dir"
foreach m $match_list {
@ -1581,7 +1581,7 @@ namespace eval punkcheck {
set punkcheck_target_relpath [file join $target_relative_to_punkcheck_dir $m]
set is_antipath 0
foreach antipath $opt_antiglob_paths {
#puts "testing file - globmatchpath $antipath vs $relative_source_path"
#puts "testing file - globmatchpath $antipath vs $relative_source_path"
if {[punk::path::globmatchpath $antipath $relative_source_path]} {
lappend antiglob_paths_matched $current_source_dir
set is_antipath 1
@ -1598,7 +1598,7 @@ namespace eval punkcheck {
#puts stdout " rel_target: $punkcheck_target_relpath"
set fetch_filerec_result [punkcheck::recordlist::get_file_record $punkcheck_target_relpath $punkcheck_records]
#change to use extract_or_create_fileset_record ?
set existing_filerec_posn [dict get $fetch_filerec_result position]
@ -1614,7 +1614,7 @@ namespace eval punkcheck {
set filerec [dict get $fetch_filerec_result record]
}
set filerec [punkcheck::recordlist::file_record_set_defaults $filerec]
#new INSTALLREC must be tagged as INSTALL-INPROGRESS to use recordlist::installfile_ method
set new_install_record [dict create tag INSTALL-INPROGRESS -tsiso $ts_start_iso -ts $ts_start -installer $opt_installer -eventid $punkcheck_eventid]
dict lappend filerec body $new_install_record ;#can't use recordlist::file_record_add_installrecord as '*-INPROGRESS' isn't a final tag - so pruning would be mucked up. No need to prune now anyway.
@ -1630,7 +1630,7 @@ namespace eval punkcheck {
#different volume or root
}
#Note this isn't a recordlist function - so it doesn't purely operate on the records
#this hits the filesystem for the sourcepath - gets checksums/timestamps depending on config.
#this hits the filesystem for the sourcepath - gets checksums/timestamps depending on config.
#It doesn't save to .punkcheck (the only punkcheck::installfile_ method which doesn't)
set filerec [punkcheck::installfile_add_source_and_fetch_metadata $punkcheck_folder $relative_source_path $filerec]
@ -1697,7 +1697,7 @@ namespace eval punkcheck {
} else {
#either cksum is different or we were unable to verify the record. Either way we can't know if the target is in sync so we must skip it
set is_skip 1
puts stderr "Skipping file copy $m target $current_target_dir/$m - require synced_target to overwrite - current target cksum compared to previous install: $target_cksum_compare"
puts stderr "Skipping file copy $m target $current_target_dir/$m - require synced_target to overwrite - current target cksum compared to previous install: $target_cksum_compare"
lappend files_skipped $current_source_dir/$m
}
} else {
@ -1728,7 +1728,7 @@ namespace eval punkcheck {
#if {$store_source_cksums} {
#}
set install_records [dict get $filerec body]
set install_records [dict get $filerec body]
set current_install_record [lindex $install_records end]
#change the tag from *-INPROGRESS to INSTALL-RECORD/SKIPPED
if {$is_skip} {
@ -1790,7 +1790,7 @@ namespace eval punkcheck {
set relative_source_path [file join $relative_source_dir $d]
set is_antipath 0
foreach antipath $opt_antiglob_paths {
#puts "testing folder - globmatchpath $antipath vs $relative_source_path"
#puts "testing folder - globmatchpath $antipath vs $relative_source_path"
if {[punk::path::globmatchpath $antipath $relative_source_path]} {
lappend antiglob_paths_matched [file join $current_source_dir $d]
#puts stdout "SKIPPING FOLDER $relative_source_path due to antiglob_path-match: $antipath "
@ -1801,11 +1801,11 @@ namespace eval punkcheck {
if {$is_antipath} {
continue
}
#if {![file exists $current_target_dir/$d]} {
# file mkdir $current_target_dir/$d
#}
set sub_opts_1 [list\
-call-depth-internal [expr {$CALLDEPTH + 1}]\
@ -1828,7 +1828,7 @@ namespace eval punkcheck {
-punkcheck_folder $punkcheck_folder\
-punkcheck_eventid $punkcheck_eventid\
-punkcheck_records $punkcheck_records\
]
]
set sub_opts [dict merge $opts $sub_opts]
set sub_result [punkcheck::install $srcdir $tgtdir {*}$sub_opts]
@ -1838,7 +1838,7 @@ namespace eval punkcheck {
lappend antiglob_paths_matched {*}[dict get $sub_result antiglob_paths_matched]
set punkcheck_records [dict get $sub_result punkcheck_records]
}
if {[string match *store* $opt_source_checksum]} {
#puts "subdirlist: $subdirlist"
if {$CALLDEPTH == 0} {
@ -1849,7 +1849,7 @@ namespace eval punkcheck {
#puts stdout ">>>>>>>>>>>>>>>>>>>"
} else {
#todo - write db INSTALLER record if -debug true
}
#puts stdout "sources_unchanged"
#puts stdout "$sources_unchanged"
@ -2108,7 +2108,7 @@ namespace eval punkcheck {
if {[dict get $file_record tag] ne "FILEINFO"} {
error "file_record_set_defaults bad file_record: tag not FILEINFO"
}
set defaults [list -keep_installrecords 3 -keep_skipped 1 -keep_inprogress 2]
set defaults [list -keep_installrecords 3 -keep_skipped 1 -keep_inprogress 2]
foreach {k v} $defaults {
if {![dict exists $file_record $k]} {
dict set file_record $k $v
@ -2186,10 +2186,10 @@ namespace eval ::punk::args::register {
# ++ +++ +++ +++ +++ +++ +++ +++ +++ +++ +++
## Ready
## Ready
package provide punkcheck [namespace eval punkcheck {
set pkg punkcheck
variable version
set version 0.1.0
set version 0.1.0
}]
return

85
src/modules/shellrun-0.1.1.tm

@ -23,7 +23,7 @@ namespace eval shellrun {
#todo - something better
if {[info exists ::punk::config::running]} {
upvar ::punk::config::running conf
set syslog_stdout [dict get $conf syslog_stdout]
set syslog_stdout [dict get $conf syslog_stdout]
set syslog_stderr [dict get $conf syslog_stderr]
set logfile_stdout [dict get $conf logfile_stdout]
set logfile_stderr [dict get $conf logfile_stderr]
@ -43,18 +43,18 @@ namespace eval shellrun {
set err [dict get [shellfilter::stack::item punksherr] device localchan]
}
namespace import ::punk::ansi::a+
namespace import ::punk::ansi::a+
namespace import ::punk::ansi::a
#repltelemetry - additional/alternative display info used in a repl context i.e info directed towards the screen
#todo - package up in repltelemetry module and rewrite proc based on whether the module was found/loaded.
#somewhat strong coupling to punk - but let's try to behave decently if it's not loaded
#The last_run_display is actually intended for the repl - but is resident in the punk namespace with a view to the possibility of a different repl being in use.
proc set_last_run_display {chunklist} {
#chunklist as understood by the
#chunklist as understood by the
if {![info exists ::punk::repltelemetry_emmitters]} {
namespace eval ::punk {
variable repltelemetry_emmitters
@ -62,7 +62,7 @@ namespace eval shellrun {
}
} else {
if {"shellrun" ni $::punk::repltelemetry_emmitters} {
lappend punk::repltelemetry_emmitters "shellrun"
lappend punk::repltelemetry_emmitters "shellrun"
}
}
@ -70,7 +70,7 @@ namespace eval shellrun {
if {[catch {llength $chunklist} errMsg]} {
error "set_last_run_display expects a list. Value supplied doesn't appear to be a well formed tcl list. '$errMsg'"
}
#todo -
#todo -
tsv::lappend repl runchunks-[tsv::get repl runid] {*}$chunklist
}
@ -140,13 +140,13 @@ namespace eval shellrun {
} else {
set nonewline 0
}
set idlist_stderr [list]
set idlist_stderr [list]
#we leave stdout without imposed ansi colouring - because the source may be colourised and because ansi-wrapping a stream at whatever boundaries it comes in at isn't a really nice thing to do.
#stderr might have source colouring - but it usually doesn't seem to, and the visual distiction of red stderr can be very handy for the run command.
#A further enhancement could be to detect well-known options such as --color and/or use a configuration for specific commands that have useful colourised stderr,
#but having an option to configure stderr to red is a compromise.
#Note that the other run commands, runout,runerr, runx don't emit in real-time - so for those commands there may be options to detect and/or post-process stdout and stderr.
#TODO - fix. This has no effect if/when the repl adds an ansiwrap transform
#TODO - fix. This has no effect if/when the repl adds an ansiwrap transform
# what we probably want to do is 'aside' that transform for runxxx commands only.
#lappend idlist_stderr [shellfilter::stack::add stderr ansiwrap -settings {-colour {red bold}}]
@ -158,7 +158,7 @@ namespace eval shellrun {
dict set callopts -debug 1
}
if {[dict exists $runoptslong --timeout]} {
dict set callopts -timeout [dict get $runoptslong --timeout] ;#convert to single dash
dict set callopts -timeout [dict get $runoptslong --timeout] ;#convert to single dash
}
#---------------------------------------------------------------------------------------------
set exitinfo [shellfilter::run $cmdargs {*}$callopts -teehandle punksh -inbuffering none -outbuffering none ]
@ -166,7 +166,7 @@ namespace eval shellrun {
foreach id $idlist_stderr {
shellfilter::stack::remove stderr $id
}
}
flush stderr
flush stdout
@ -191,10 +191,10 @@ namespace eval shellrun {
set redir ">&@stdout <@stdin"
uplevel 1 [list ::catch [concat exec $redir $new [lrange $args 1 end]] ::tcl::UnknownResult ::tcl::UnknownOptions]
#we can't detect stdout/stderr output from the exec
#for now emit an extra \n on stderr
#we can't detect stdout/stderr output from the exec
#for now emit an extra \n on stderr
#todo - there is probably no way around this but to somehow exec in the context of a completely separate console
#This is probably a tricky problem - especially to do cross-platform
#This is probably a tricky problem - especially to do cross-platform
#
# - use [dict get $::tcl::UnknownOptions -code] (0|1) exit
if {[dict get $::tcl::UnknownOptions -code] == 0} {
@ -230,9 +230,9 @@ namespace eval shellrun {
} else {
set nonewline 0
}
#puts stdout "RUNOUT cmdargs: $cmdargs"
#todo add -data boolean and -data lastwrite to -settings with default being -data all
# because sometimes we're only interested in last char (e.g to detect something was output)
@ -268,7 +268,7 @@ namespace eval shellrun {
if {"-tcl" in $runopts} {
} else {
#we must raise an error.
#we must raise an error.
#todo - check errorInfo makes sense.. return -code? tailcall?
#
set msg ""
@ -281,9 +281,10 @@ namespace eval shellrun {
set chunklist [list]
#exitcode not part of return value for runout - colourcode appropriately
set n $RST
set n $RST
set c ""
if [dict exists $exitinfo exitcode] {
if {[dict exists $exitinfo exitcode]} {
set code [dict get $exitinfo exitcode]
if {$code == 0} {
set c [a+ green]
@ -291,7 +292,7 @@ namespace eval shellrun {
set c [a+ white bold]
}
lappend chunklist [list "info" "$c$exitinfo$n"]
} elseif [dict exists $exitinfo error] {
} elseif {[dict exists $exitinfo error]} {
set c [a+ yellow bold]
lappend chunklist [list "info" "${c}error [dict get $exitinfo error]$n"]
lappend chunklist [list "info" "errorCode [dict get $exitinfo errorCode]"]
@ -330,7 +331,7 @@ namespace eval shellrun {
} else {
set o $::shellrun::runout
}
append chunk "$o"
append chunk "$o"
}
lappend chunklist [list result $chunk]
@ -347,7 +348,7 @@ namespace eval shellrun {
proc runerr {args} {
#set_last_run_display [list]
variable runout
variable runout
variable runerr
set runout ""
set runerr ""
@ -398,17 +399,15 @@ namespace eval shellrun {
set n [a]
set c ""
if [dict exists $exitinfo exitcode] {
if {[dict exists $exitinfo exitcode]} {
set code [dict get $exitinfo exitcode]
if {$code == 0} {
set c [a+ green]
} else {
set c [a+ white bold]
}
lappend chunklist [list "info" "$c$exitinfo$n"]
} elseif [dict exists $exitinfo error] {
} elseif {[dict exists $exitinfo error]} {
set c [a+ yellow bold]
lappend chunklist [list "info" "error [dict get $exitinfo error]"]
lappend chunklist [list "info" "errorCode [dict get $exitinfo errorCode]"]
@ -459,8 +458,8 @@ namespace eval shellrun {
proc runx {args} {
#set_last_run_display [list]
variable runout
#set_last_run_display [list]
variable runout
variable runerr
set runout ""
set runerr ""
@ -491,7 +490,7 @@ namespace eval shellrun {
set stderr_stackid [shellfilter::stack::add stderr var -action float-locked -junction 1 -settings {-varname ::shellrun::runerr}]
set stdout_stackid [shellfilter::stack::add stdout var -action float-locked -junction 1 -settings {-varname ::shellrun::runout}]
}
set callopts ""
if {"-tcl" in $runopts} {
append callopts " -tclscript 1"
@ -505,7 +504,7 @@ namespace eval shellrun {
flush stderr
flush stdout
if {[dict exists $exitinfo error]} {
if {"-tcl" in $runopts} {
@ -514,7 +513,7 @@ namespace eval shellrun {
error [dict get $exitinfo error]
}
}
#set x [shellfilter::stack::add stdout var -action sink-locked -settings {-varname ::repl::runxoutput}]
set chunk ""
@ -568,7 +567,7 @@ namespace eval shellrun {
set exitdict [list exitcode $code]
} elseif {[dict exists $exitinfo result]} {
# presumably from a -tcl call
set val [dict get $exitinfo result]
set val [dict get $exitinfo result]
lappend chunklist [list "info" " "]
lappend chunklist [list "result" result]
lappend chunklist [list "info" result]
@ -626,15 +625,15 @@ namespace eval shellrun {
#we can only call runraw with a single (presumably braced) string if we want to use it from both repl and tcl scripts (why? todo with unbalanced quotes/braces?)
proc runraw {commandline} {
#runraw fails as intended - because we can't bypass exec/open interference quoting :/
#set_last_run_display [list]
variable runout
#set_last_run_display [list]
variable runout
variable runerr
set runout ""
set runerr ""
#return [shellfilter::run [lrange $args 1 end] -teehandle punksh -inbuffering none -outbuffering none -stdinhandler ::repl::repl_handler]
puts stdout ">>runraw got: $commandline"
#run always echoes anyway.. as we aren't diverting stdout/stderr off for capturing
#for consistency with other runxxx commands - we'll just consume it. (review)
@ -666,14 +665,14 @@ namespace eval shellrun {
}
}
}
puts stdout ">>runraw runwords: $runwords"
set runwords [lrange $runwords 1 end]
puts stdout ">>runraw runwords: $runwords"
#set args [lrange $args 1 end]
#set runwords [lrange $wordparts 1 end]
set known_runopts [list "-echo" "-e" "-terminal" "-t"]
set aliases [list "-e" "-echo" "-echo" "-echo" "-t" "-terminal" "-terminal" "-terminal"] ;#include map to self
set runopts [list]
@ -681,17 +680,17 @@ namespace eval shellrun {
set idx_first_cmdarg [lsearch -not $runwords "-*"]
set runopts [lrange $runwords 0 $idx_first_cmdarg-1]
set cmdwords [lrange $runwords $idx_first_cmdarg end]
foreach o $runopts {
if {$o ni $known_runopts} {
error "runraw: Unknown runoption $o"
}
}
set runopts [lmap o $runopts {dict get $aliases $o}]
set cmd_as_string [join $cmdwords " "]
puts stdout ">>cmd_as_string: $cmd_as_string"
if {"-terminal" in $runopts} {
#fake terminal using 'script' command.
#not ideal: smushes stdout & stderr together amongst other problems
@ -702,7 +701,7 @@ namespace eval shellrun {
} else {
set exitinfo [shellfilter::run $cmdwords -teehandle punksh -inbuffering line -outbuffering none ]
}
if {[dict exists $exitinfo error]} {
#todo - check errorInfo makes sense.. return -code? tailcall?
error [dict get $exitinfo error]
@ -764,7 +763,7 @@ namespace eval shellrun {
interp alias {} ro {} shellrun::runout
interp alias {} re {} shellrun::runerr
interp alias {} rx {} shellrun::runx
}
@ -772,7 +771,7 @@ namespace eval shellrun {
proc test_cffi {} {
package require test_cffi
cffi::Wrapper create ::shellrun::kernel32 [file join $env(windir) system32 Kernel32.dll]
::shellrun::kernel32 stdcall CreateProcessA
::shellrun::kernel32 stdcall CreateProcessA
#todo - stuff.
return ::shellrun::kernel32
}

106
src/modules/shellthread-1.6.1.tm

@ -49,7 +49,7 @@ namespace eval shellthread::worker {
variable logfile
variable settings
interp bgerror {} shellthread::worker::bgerror
#package require overtype ;#overtype uses tcllib textutil, punk::char etc - currently too heavyweight in terms of loading time for use in threads.
#package require overtype ;#overtype uses tcllib textutil, punk::char etc - currently too heavyweight in terms of loading time for use in threads.
variable client_ids
variable ts_start_micros
lappend client_ids $tidclient
@ -108,7 +108,7 @@ namespace eval shellthread::worker {
chan configure $readchan -translation lf
if {$readchan ni [chan names]} {
error "shellthread::worker::start_pipe_read - inpipe not configured. Use shellthread::manager::set_pipe_read_from_client to thread::transfer the pipe end"
error "shellthread::worker::start_pipe_read - inpipe not configured. Use shellthread::manager::set_pipe_read_from_client to thread::transfer the pipe end"
}
set inpipe $readchan
chan configure $readchan -blocking 0
@ -123,15 +123,15 @@ namespace eval shellthread::worker {
set chunksize [chan gets $chan chunk]
if {$chunksize >= 0} {
if {![chan eof $chan]} {
::shellthread::worker::log pipe 0 - $source - info $chunk\n $writebuffering
::shellthread::worker::log pipe 0 - $source - info $chunk\n $writebuffering
} else {
::shellthread::worker::log pipe 0 - $source - info $chunk $writebuffering
::shellthread::worker::log pipe 0 - $source - info $chunk $writebuffering
}
}
} else {
set chunk [chan read $chan]
::shellthread::worker::log pipe 0 - $source - info $chunk $writebuffering
}
::shellthread::worker::log pipe 0 - $source - info $chunk $writebuffering
}
if {[chan eof $chan]} {
chan event $chan readable {}
set $waitfor "pipe"
@ -143,10 +143,10 @@ namespace eval shellthread::worker {
variable outpipe
set defaults [dict create -buffering \uFFFF ]
set opts [dict merge $defaults $args]
#todo!
set readchan stdin
if {[dict exists $opts -readbuffering]} {
set readbuffering [dict get $opts -readbuffering]
} else {
@ -168,15 +168,15 @@ namespace eval shellthread::worker {
can configure $writechan -buffering $writebuffering
}
}
if {$writechan ni [chan names]} {
error "shellthread::worker::start_pipe_write - outpipe not configured. Use shellthread::manager::set_pipe_write_to_client to thread::transfer the pipe end"
error "shellthread::worker::start_pipe_write - outpipe not configured. Use shellthread::manager::set_pipe_write_to_client to thread::transfer the pipe end"
}
set outpipe $writechan
chan configure $readchan -blocking 0
chan configure $writechan -blocking 0
set waitvar ::shellthread::worker::wait($outpipe,[clock micros])
chan event $readchan readable [list apply {{chan writechan source waitfor readbuffering} {
if {$readbuffering eq "line"} {
set chunksize [chan gets $chan chunk]
@ -194,7 +194,7 @@ namespace eval shellthread::worker {
if {[chan eof $chan]} {
chan event $chan readable {}
set $waitfor "pipe"
chan close $writechan
chan close $writechan
if {$chan ne "stdin"} {
chan close $chan
}
@ -209,18 +209,18 @@ namespace eval shellthread::worker {
variable sysloghost_port
variable sock
if {[string length $sysloghost_port]} {
if {[catch {fconfigure $sock} state]} {
if {[catch {chan configure $sock} state]} {
set sock [udp_open]
fconfigure $sock -buffering none -translation binary
fconfigure $sock -remote $sysloghost_port
chan configure $sock -buffering none -translation binary
chan configure $sock -remote $sysloghost_port
}
}
}
}
proc _reconnect {} {
variable sock
catch {close $sock}
_initsock
return [fconfigure $sock]
return [chan configure $sock]
}
proc send_info {client_tid ts_sent source msg} {
@ -242,12 +242,12 @@ namespace eval shellthread::worker {
set tail_crlf 0
set tail_lf 0
set tail_cr 0
#for cooked - always remove the trailing newline before splitting..
#for cooked - always remove the trailing newline before splitting..
#
#note that if we got our data from reading a non-line-buffered binary channel - then this naive line splitting will not split neatly for mixed line-endings.
#
#Possibly not critical as cooked is for logging and we are still preserving all \r and \n chars - but review and consider implementing a better split
#but add it back exactly as it was afterwards
#but add it back exactly as it was afterwards
#we can always split on \n - and any adjacent \r will be preserved in the rejoin
set lastchar [string range $logchunk end end]
if {[string range $logchunk end-1 end] eq "\r\n"} {
@ -283,9 +283,9 @@ namespace eval shellthread::worker {
#set col0 [string repeat " " 9]
#set col1 [string repeat " " 27]
#set col2 [string repeat " " 11]
#set col3 [string repeat " " 22]
#set col3 [string repeat " " 22]
##do not columnize the final data column or append to tail - or we could muck up the crlf integrity
#lassign [list [overtype::left $col0 $idtail] [overtype::left $col1 $time_info] [overtype::left $col2 $lagfp] [overtype::left $col3 $source]] c0 c1 c2 c3
#lassign [list [overtype::left $col0 $idtail] [overtype::left $col1 $time_info] [overtype::left $col2 $lagfp] [overtype::left $col3 $source]] c0 c1 c2 c3
set w0 9
set w1 27
@ -297,15 +297,15 @@ namespace eval shellthread::worker {
[format %-${w1}s $time_info]\
[format %-${w2}s $lagfp]\
[format %-${w3}s $source]\
] c0 c1 c2 c3
] c0 c1 c2 c3
set c2_blank [string repeat " " $w2]
#split on \n no matter the actual line-ending in use
#shouldn't matter as long as we don't add anything at the end of the line other than the raw data
#ie - don't quote or add spaces
set lines [split $logchunk \n]
set lines [split $logchunk \n]
set i 1
set outlines [list]
foreach ln $lines {
@ -324,13 +324,13 @@ namespace eval shellthread::worker {
set logchunk "[join $outlines \r]\r"
} else {
#no trailing linefeed
set logchunk [join $outlines \n]
set logchunk [join $outlines \n]
}
#set logchunk "[overtype::left $col0 $idtail] [overtype::left $col1 $time_info] [overtype::left $col2 "+$lagfp"] [overtype::left $col3 $source] $msg"
}
if {[string length $sysloghost_port]} {
_initsock
catch {puts -nonewline $sock $logchunk}
@ -348,7 +348,7 @@ namespace eval shellthread::worker {
}
}
# - withdraw just this client
# - withdraw just this client
proc finish {tidclient} {
variable client_ids
if {($tidclient in $clientids) && ([llength $clientids] == 1)} {
@ -373,11 +373,11 @@ namespace eval shellthread::worker {
#however.. how can we set a timeout on a thread::join ?
#by telling the thread to release itself - we can wait on the thread::send variable
# This needs review - because it's unclear that -wait even works on self
# (what does it mean to wait for the target thread to exit if the target is self??)
# (what does it mean to wait for the target thread to exit if the target is self??)
thread::release -wait
return [thread::id]
return [thread::id]
} else {
return ""
return ""
}
}
@ -388,7 +388,7 @@ namespace eval shellthread::worker {
namespace eval shellthread::manager {
variable workers [dict create]
variable worker_errors [list]
variable timeouts
variable timeouts
variable free_threads [list]
#variable log_threads
@ -401,7 +401,7 @@ namespace eval shellthread::manager {
if {[tcl::dict::exists $dictValue {*}$keys]} {
return [tcl::dict::get $dictValue {*}$keys]
} else {
return [lindex $args end]
return [lindex $args end]
}
}
#new datastructure regarding workers and sourcetags required.
@ -412,7 +412,7 @@ namespace eval shellthread::manager {
#If the thread which started the thread calls leave_worker with that 'primary' sourcetag it means others won't be able to use that target - which seems reasonable.
#If another thread want's to maintain joinability beyond the span provided by the starting client,
#it can join with both the primary tag and a tag it will actually use for logging.
#A thread can join the logger with any existingtag - not just the 'primary'
#A thread can join the logger with any existingtag - not just the 'primary'
#(which is arbitrary anyway. It will usually be the first in the list - but may be unsubscribed by clients and disappear)
proc join_worker {existingtag sourcetaglist} {
set client_tid [thread::id]
@ -431,15 +431,15 @@ namespace eval shellthread::manager {
#it is up to caller to use a unique sourcetag (e.g by prefixing with own thread::id etc)
# This allows multiple threads to more easily write to the same named sourcetag if necessary
# todo - change sourcetag for a list of tags which will be handled by the same thread. e.g for multiple threads logging to same file
# todo - change sourcetag for a list of tags which will be handled by the same thread. e.g for multiple threads logging to same file
#
# todo - some protection mechanism for case where target is a file to stop creation of multiple worker threads writing to same file.
# Even if we use open fd,close fd wrapped around writes.. it is probably undesirable to have multiple threads with same target
# Even if we use open fd,close fd wrapped around writes.. it is probably undesirable to have multiple threads with same target
# On the other hand socket targets such as UDP can happily be written to by multiple threads.
# For now the mechanism is that a call to new_worker (rename to open_worker?) will join the same thread if a sourcetag matches..
# For now the mechanism is that a call to new_worker (rename to open_worker?) will join the same thread if a sourcetag matches.
# but, as sourcetags can get removed(unsubbed via leave_worker) this doesn't guarantee two threads with same -file settings won't fight.
# Also.. the settingsdict is ignored when joining with a tag that exists.. this is problematic.. e.g logrotation where previous file still being written by existing worker
# todo - rename 'sourcetag' concept to 'targettag' ?? the concept is a mixture of both.. it is somewhat analagous to a syslog 'facility'
# todo - rename 'sourcetag' concept to 'targettag' ?? the concept is a mixture of both.. it is somewhat analagous to a syslog 'facility'
# probably new_worker should disallow auto-joining and we allow different workers to handle same tags simultaneously to support overlap during logrotation etc.
proc new_worker {sourcetaglist {settingsdict {}}} {
variable workers
@ -455,7 +455,7 @@ namespace eval shellthread::manager {
set workertype [string tolower [dict get $settingsdict -workertype]]
set known_workertypes [list pipe message]
if {$workertype ni $known_workertypes} {
error "new_worker - unknown -workertype $workertype. Expected one of '$known_workertypes'"
error "new_worker - unknown -workertype $workertype. Expected one of '$known_workertypes'"
}
if {[dict exists $workers $sourcetag]} {
@ -502,8 +502,8 @@ namespace eval shellthread::manager {
#if {$tcllib ni $::auto_path} {
# lappend ::auto_path $tcllib
#}
set ::settingsinfo [dict create %sd%]
set ::settingsinfo [dict create %sd%]
#if the executable running things is something like a tclkit,
# then it's likely we will need to use the caller's auto_path and tcl::tm::list to find things
#The caller can tune the thread's package search by providing a settingsdict
@ -573,7 +573,7 @@ namespace eval shellthread::manager {
}
proc write_log {source msg args} {
variable workers
variable workers
set ts_micros_sent [clock micros]
set defaults [list -async 1 -level info]
set opts [dict merge $defaults $args]
@ -584,12 +584,12 @@ namespace eval shellthread::manager {
return
}
if {![thread::exists $tidworker]} {
# -syslog -file ?
# -syslog -file ?
set tidworker [new_worker $source]
}
} else {
#auto create with no requirement to call new_worker.. warn?
# -syslog -file ?
# -syslog -file ?
error "write_log no log opened for source: $source"
set tidworker [new_worker $source]
}
@ -599,7 +599,7 @@ namespace eval shellthread::manager {
} else {
thread::send $tidworker [list ::shellthread::worker::send_info $client_tid $ts_micros_sent $source $msg]
}
}
}
proc report_worker_errors {errdict} {
variable workers
set reporting_tid [dict get $errdict worker_tid]
@ -641,7 +641,7 @@ namespace eval shellthread::manager {
set shuttingdown_workers [list]
foreach deadtag $subscriberless_tags {
set workertid [dict get $workers $deadtag tid]
set worker_tags [get_worker_tagstate $workertid]
set worker_tags [get_worker_tagstate $workertid]
set subscriber_count 0
set kill_count 0 ;#number of ts_end_list entries - even one indicates thread is doomed
foreach taginfo $worker_tags {
@ -690,8 +690,8 @@ namespace eval shellthread::manager {
if {[info exists timeoutarr(shutdown_free_threads)]} {
#already called
return false
}
#set timeoutarr(shutdown_free_threads) waiting
}
#set timeoutarr(shutdown_free_threads) waiting
#after $timeout [list set timeoutarr(shutdown_free_threads) timed-out]
set ::shellthread::waitfor waiting
after $timeout [list set ::shellthread::waitfor]
@ -708,7 +708,7 @@ namespace eval shellthread::manager {
}
if {[llength $waiting_for]} {
for {set i 0} {$i < [llength $waiting_for]} {incr i} {
vwait ::shellthread::waitfor
vwait ::shellthread::waitfor
if {$::shellthread::waitfor eq "timed-out"} {
set timedout 1
break
@ -724,9 +724,9 @@ namespace eval shellthread::manager {
#TODO - important.
#REVIEW!
#since moving to the unsubscribe mechansm - close_worker $source isn't being called
# - we need to set a limit to the number of free threads and shut down excess when detected during unsubscription
#instruction to shut-down the thread that has this source.
#instruction to shut-down the thread that has this source.
# - we need to set a limit to the number of free threads and shut down excess when detected during unsubscription
#instruction to shut-down the thread that has this source.
#instruction to shut-down the thread that has this source.
proc close_worker {source {timeout 2500}} {
variable workers
variable worker_errors
@ -751,7 +751,7 @@ namespace eval shellthread::manager {
set ts_end_list [dict get $workers $source ts_end_list] ;#ts_end_list is just a list of timestamps of closing calls for this source - only one is needed to close, but they may all come in a flurry.
if {[llength $ts_end_list]} {
set last_end_ts [lindex $ts_end_list end]
if {[expr {(($tsnow - $last_end_ts) / 1000) >= $timeout}]} {
if {(($tsnow - $last_end_ts) / 1000) >= $timeout} {
lappend ts_end_list $ts_now
dict set workers $source ts_end_list $ts_end_list
} else {
@ -773,7 +773,7 @@ namespace eval shellthread::manager {
#thread::send -async $tidworker [string map [list %tidclient% [thread::id]] {
# shellthread::worker::terminate %tidclient%
#}] timeoutarr($source)
vwait timeoutarr($source)
#puts stderr "shellthread::manager::close_worker: thread $tidworker for source $source DONE1"

6
src/modules/tcl9test-999999.0a1.0.tm

@ -53,7 +53,7 @@ namespace eval [lassign [split [file rootname [file tail [info script] ]] -] pkg
uplevel #0 [list package provide $pkgtail $version]
uplevel #0 [list package provide $pkgtail $version]
#package provide [lassign {tcl9test 999999.0a1.0} pkg ver]$pkg [namespace eval $pkg[set pkg {}] {list [variable version $::ver[set ::ver {}]]$version}]
}
@ -64,9 +64,9 @@ namespace eval [lassign [split [file rootname [file tail [info script] ]] -] pkg
#package provide [lassign {tcl9test 999999.0a1.0} pkg ver]$pkg [namespace eval $pkg[set pkg {}] {list [variable version $::ver[set ::ver {}]]$version}]
# ++ +++ +++ +++ +++ +++ +++ +++ +++ +++ +++
## Ready
## Ready
#package provide tcl9test [namespace eval tcl9test {
# variable version
# set version 999999.0a1.0
# set version 999999.0a1.0
#}]
#return

936
src/modules/textblock-999999.0a1.0.tm

File diff suppressed because it is too large Load Diff
Loading…
Cancel
Save