@@ -172,31 +172,40 @@ class _ChRootContext(typing.ContextManager[None]):
172
172
:type conn: ExecHelper
173
173
:param path: chroot path or None for no chroot
174
174
:type path: str | pathlib.Path | None
175
- :raises TypeError: incorrect type of path variable
175
+ :param chroot_exe: chroot executable
176
+ :type chroot_exe: str
177
+ :raises TypeError: incorrect type of path or chroot_exe variable
176
178
177
179
.. versionadded:: 4.1.0
178
180
"""
179
181
180
- __slots__ = ("_chroot_status" , "_conn" , "_path" )
182
+ __slots__ = ("_chroot_status" , "_chroot_exe_status" , " _conn" , "_path" , "_exe " )
181
183
182
- def __init__ (self , conn : ExecHelper , path : ChRootPathSetT = None ) -> None :
184
+ def __init__ (self , conn : ExecHelper , path : ChRootPathSetT = None , chroot_exe : str = "chroot" ) -> None :
183
185
"""Context manager for call commands with sudo.
184
186
185
- :raises TypeError: incorrect type of path variable
187
+ :raises TypeError: incorrect type of path or chroot_exe variable
186
188
"""
187
189
self ._conn : ExecHelper = conn
188
190
self ._chroot_status : str | None = conn ._chroot_path
191
+ self ._chroot_exe_status : str = conn ._chroot_exe
189
192
if path is None or isinstance (path , str ):
190
193
self ._path : str | None = path
191
194
elif isinstance (path , pathlib .Path ):
192
195
self ._path = path .as_posix () # get absolute path
193
196
else :
194
197
raise TypeError (f"path={ path !r} is not instance of { ChRootPathSetT } " )
198
+ if isinstance (chroot_exe , str ):
199
+ self ._exe : str = chroot_exe
200
+ else :
201
+ raise TypeError (f"chroot_exe={ chroot_exe !r} is not instance of str" )
195
202
196
203
def __enter__ (self ) -> None :
197
204
self ._conn .__enter__ ()
198
205
self ._chroot_status = self ._conn ._chroot_path
199
206
self ._conn ._chroot_path = self ._path
207
+ self ._chroot_exe_status = self ._conn ._chroot_exe
208
+ self ._conn ._chroot_exe = self ._exe
200
209
201
210
def __exit__ (
202
211
self ,
@@ -205,6 +214,7 @@ def __exit__(
205
214
exc_tb : TracebackType | None ,
206
215
) -> None :
207
216
self ._conn ._chroot_path = self ._chroot_status
217
+ self ._conn ._chroot_exe = self ._chroot_exe_status
208
218
self ._conn .__exit__ (exc_type , exc_val , exc_tb )
209
219
210
220
@@ -224,10 +234,12 @@ class ExecHelper(
224
234
.. versionchanged:: 1.2.0 log_mask_re regex rule for masking cmd
225
235
.. versionchanged:: 1.3.5 make API public to use as interface
226
236
.. versionchanged:: 4.1.0 support chroot
237
+ .. versionchanged:: 8.1.0 support chroot_exe
227
238
"""
228
239
229
240
__slots__ = (
230
241
"__chroot_path" ,
242
+ "__chroot_exe" ,
231
243
"__context_count" ,
232
244
"__lock" ,
233
245
"__logger" ,
@@ -240,6 +252,7 @@ def __init__(self, log_mask_re: LogMaskReT = None, *, logger: logging.Logger) ->
240
252
self .__logger : logging .Logger = logger
241
253
self .log_mask_re : LogMaskReT = log_mask_re
242
254
self .__chroot_path : str | None = None
255
+ self .__chroot_exe : str = "chroot"
243
256
self .__context_count = 0
244
257
245
258
@property
@@ -292,18 +305,51 @@ def _chroot_path(self) -> None:
292
305
"""
293
306
self .__chroot_path = None
294
307
295
- def chroot (self , path : ChRootPathSetT ) -> _ChRootContext :
308
+ @property
309
+ def _chroot_exe (self ) -> str :
310
+ """Exe for chroot
311
+
312
+ :rtype: str
313
+ .. versionadded:: 8.1.0
314
+ """
315
+ return self .__chroot_exe
316
+
317
+ @_chroot_exe .setter
318
+ def _chroot_exe (self , new_state : str ) -> None :
319
+ """Executable for chroot if set.
320
+
321
+ :param new_state: new exe
322
+ :type new_state: str
323
+ :raises TypeError: Not supported exe information
324
+ .. versionadded:: 8.1.0
325
+ """
326
+ if isinstance (new_state , str ):
327
+ self .__chroot_exe = new_state
328
+ else :
329
+ raise TypeError (f"chroot_exe is expected to be string, but set { new_state !r} " )
330
+
331
+ @_chroot_exe .deleter
332
+ def _chroot_exe (self ) -> None :
333
+ """Restore chroot executable.
334
+
335
+ .. versionadded:: 8.1.0
336
+ """
337
+ self .__chroot_exe = "chroot"
338
+
339
+ def chroot (self , path : ChRootPathSetT , chroot_exe : str = "chroot" ) -> _ChRootContext :
296
340
"""Context manager for changing chroot rules.
297
341
298
342
:param path: chroot path or none for working without chroot.
299
343
:type path: str | pathlib.Path | None
344
+ :param chroot_exe: chroot exe
345
+ :type chroot_exe: str
300
346
:return: context manager with selected chroot state inside
301
347
:rtype: typing.ContextManager
302
348
303
349
.. Note:: Enter and exit main context manager is produced as well.
304
350
.. versionadded:: 4.1.0
305
351
"""
306
- return _ChRootContext (conn = self , path = path )
352
+ return _ChRootContext (conn = self , path = path , chroot_exe = chroot_exe )
307
353
308
354
@property
309
355
def _context_count (self ) -> int :
@@ -348,7 +394,7 @@ def _mask_command(self, cmd: str, log_mask_re: LogMaskReT = None) -> str:
348
394
349
395
return _helpers .mask_command (cmd .rstrip (), self .log_mask_re , log_mask_re )
350
396
351
- def _prepare_command (self , cmd : str , chroot_path : str | None = None ) -> str :
397
+ def _prepare_command (self , cmd : str , chroot_path : str | None = None , chroot_exe : str = "chroot" ) -> str :
352
398
"""Prepare command: cower chroot and other cases.
353
399
354
400
:param cmd: main command
@@ -358,7 +404,7 @@ def _prepare_command(self, cmd: str, chroot_path: str | None = None) -> str:
358
404
:return: final command, includes chroot, if required
359
405
:rtype: str
360
406
"""
361
- return _helpers .chroot_command (cmd , chroot_path = chroot_path or self ._chroot_path )
407
+ return _helpers .chroot_command (cmd , chroot_path = chroot_path or self ._chroot_path , chroot_exe = chroot_exe )
362
408
363
409
@abc .abstractmethod
364
410
def _exec_command (
@@ -427,6 +473,7 @@ def open_execute_context(
427
473
open_stdout : bool = True ,
428
474
open_stderr : bool = True ,
429
475
chroot_path : str | None = None ,
476
+ chroot_exe : str = "chroot" ,
430
477
** kwargs : typing .Any ,
431
478
) -> ExecuteContext :
432
479
"""Get execution context manager.
@@ -441,6 +488,8 @@ def open_execute_context(
441
488
:type open_stderr: bool
442
489
:param chroot_path: chroot path override
443
490
:type chroot_path: str | None
491
+ :param chroot_exe: chroot exe override
492
+ :type chroot_exe: str
444
493
:param kwargs: additional parameters for call.
445
494
:type kwargs: typing.Any
446
495
@@ -460,6 +509,7 @@ def execute(
460
509
open_stderr : bool = True ,
461
510
log_stderr : bool = True ,
462
511
chroot_path : str | None = None ,
512
+ chroot_exe : str = "chroot" ,
463
513
** kwargs : typing .Any ,
464
514
) -> exec_result .ExecResult :
465
515
"""Execute command and wait for return code.
@@ -485,6 +535,8 @@ def execute(
485
535
:type log_stderr: bool
486
536
:param chroot_path: chroot path override
487
537
:type chroot_path: str | None
538
+ :param chroot_exe: chroot exe override
539
+ :type chroot_exe: str
488
540
:param kwargs: additional parameters for call.
489
541
:type kwargs: typing.Any
490
542
:return: Execution result
@@ -495,6 +547,7 @@ def execute(
495
547
.. versionchanged:: 2.1.0 Allow parallel calls
496
548
.. versionchanged:: 7.0.0 Allow command as list of arguments. Command will be joined with components escaping.
497
549
.. versionchanged:: 8.0.0 chroot path exposed.
550
+ .. versionchanged:: 8.1.0 chroot_exe added.
498
551
"""
499
552
log_level : int = logging .INFO if verbose else logging .DEBUG
500
553
cmd = _helpers .cmd_to_string (command )
@@ -511,6 +564,7 @@ def execute(
511
564
open_stdout = open_stdout ,
512
565
open_stderr = open_stderr ,
513
566
chroot_path = chroot_path ,
567
+ chroot_exe = chroot_exe ,
514
568
** kwargs ,
515
569
) as async_result :
516
570
result : exec_result .ExecResult = self ._exec_command (
@@ -540,6 +594,7 @@ def __call__(
540
594
open_stderr : bool = True ,
541
595
log_stderr : bool = True ,
542
596
chroot_path : str | None = None ,
597
+ chroot_exe : str = "chroot" ,
543
598
** kwargs : typing .Any ,
544
599
) -> exec_result .ExecResult :
545
600
"""Execute command and wait for return code.
@@ -565,6 +620,8 @@ def __call__(
565
620
:type log_stderr: bool
566
621
:param chroot_path: chroot path override
567
622
:type chroot_path: str | None
623
+ :param chroot_exe: chroot exe override
624
+ :type chroot_exe: str
568
625
:param kwargs: additional parameters for call.
569
626
:type kwargs: typing.Any
570
627
:return: Execution result
@@ -584,6 +641,7 @@ def __call__(
584
641
open_stderr = open_stderr ,
585
642
log_stderr = log_stderr ,
586
643
chroot_path = chroot_path ,
644
+ chroot_exe = chroot_exe ,
587
645
** kwargs ,
588
646
)
589
647
0 commit comments