Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Release 0.3.1 #80

Merged
merged 5 commits into from
Dec 18, 2023
Merged
Show file tree
Hide file tree
Changes from 2 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 0 additions & 1 deletion Benchmark/System/Process.hs
Original file line number Diff line number Diff line change
Expand Up @@ -23,7 +23,6 @@ import qualified Streamly.Internal.System.Command as Cmd

-- Internal imports
import qualified Streamly.Internal.FileSystem.Handle as FH
import qualified Streamly.Internal.System.Process as Proc

-- XXX replace with streamly versions once they are fixed
{-# INLINE rights #-}
Expand Down
4 changes: 2 additions & 2 deletions src/DocTestCommand.hs
Original file line number Diff line number Diff line change
Expand Up @@ -14,7 +14,7 @@

For APIs that have not been released yet.

>>> import qualified Streamly.Internal.Console.Stdio as Stdio
>>> import qualified Streamly.Internal.FileSystem.Dir as Dir
>>> import qualified Streamly.Internal.Console.Stdio as Stdio (putBytes, putChars, putChunks)
>>> import qualified Streamly.Internal.FileSystem.Dir as Dir (readFiles)
>>> import qualified Streamly.Internal.System.Process as Process
-}
7 changes: 3 additions & 4 deletions src/DocTestProcess.hs
Original file line number Diff line number Diff line change
Expand Up @@ -13,9 +13,8 @@

For APIs that have not been released yet.

>>> import qualified Streamly.Internal.Console.Stdio as Stdio
>>> import qualified Streamly.Internal.Data.Stream as Stream
>>> import qualified Streamly.Internal.FileSystem.Dir as Dir
>>> import qualified Streamly.Internal.Console.Stdio as Stdio (putChars, putChunks)
>>> import qualified Streamly.Internal.FileSystem.Dir as Dir (readFiles)
>>> import qualified Streamly.Internal.System.Process as Process
>>> import qualified Streamly.Internal.Unicode.Stream as Unicode
>>> import qualified Streamly.Internal.Unicode.Stream as Unicode (lines)
-}
4 changes: 2 additions & 2 deletions src/Streamly/Internal/System/Command.hs
Original file line number Diff line number Diff line change
Expand Up @@ -53,7 +53,7 @@ import qualified Streamly.Internal.System.Process as Process

#include "DocTestCommand.hs"

-- Posix compliant quote excaping:
-- | Posix compliant quote escaping:
--
-- $ echo 'hello\\"world'
-- hello\\"world
Expand Down Expand Up @@ -169,7 +169,7 @@ pipeWith f cmd input =
-- If the input stream throws an exception or if the output stream is garbage
-- collected before it could finish then the process is terminated with SIGTERM.
--
-- If the process terminates with a non-zero exit code then a 'ProcessFailure'
-- If the process terminates with a non-zero exit code then a 'Process.ProcessFailure'
-- exception is raised.
--
-- The following code is equivalent to the shell command @echo "hello world" |
Expand Down
118 changes: 90 additions & 28 deletions src/Streamly/Internal/System/Process.hs
Original file line number Diff line number Diff line change
Expand Up @@ -60,17 +60,18 @@ module Streamly.Internal.System.Process
-}
, closeFiles
, newProcessGroup
, Session (..)
, setSession

-- * Posix Only Options
-- | These options have no effect on Windows.
, parentIgnoresInterrupt
, interruptChildOnly
, setUserId
, setGroupId

-- * Windows Only Options
-- | These options have no effect on Posix.
, waitForChildTree
, waitForDescendants

-- * Internal
, inheritStdin
Expand Down Expand Up @@ -111,11 +112,14 @@ module Streamly.Internal.System.Process
, pipeChunksEitherWith

-- * Standalone Processes
, standalone
, interactive
, foreground
, daemon
, standalone

-- * Deprecated
, parentIgnoresInterrupt
, waitForChildTree
, interactive
, processBytes
, processChunks
)
Expand Down Expand Up @@ -326,7 +330,9 @@ closeFiles x (Config cfg) = Config $ cfg { close_fds = x }
-- | If 'True' the new process starts a new process group, becomes a process
-- group leader, its pid becoming the process group id.
--
-- See the POSIX @setpgid@ man page.
-- See the POSIX
-- <https://man7.org/linux/man-pages/man2/setpgid.2.html setpgid>
-- man page.
--
-- Default is 'False', the new process belongs to the parent's process group.
newProcessGroup :: Bool -> Config -> Config
Expand All @@ -336,7 +342,9 @@ newProcessGroup x (Config cfg) = Config $ cfg { create_group = x }
-- parent process. This is the default.
--
-- 'NewSession' makes the new process start with a new session without a
-- controlling terminal. On POSIX, @setsid@ is used to create a new process
-- controlling terminal. On POSIX,
-- <https://man7.org/linux/man-pages/man2/setsid.2.html setsid>
-- is used to create a new process
-- group and session, the pid of the new process is the session id and process
-- group id as well. On Windows @DETACHED_PROCESS@ flag is used to detach the
-- process from inherited console session.
Expand All @@ -346,10 +354,12 @@ newProcessGroup x (Config cfg) = Config $ cfg { create_group = x }
-- nothing.
--
-- For Windows see
--
-- * https://learn.microsoft.com/en-us/windows/win32/procthread/process-creation-flags
-- * https://learn.microsoft.com/en-us/windows/console/creation-of-a-console .
--
-- For POSIX see, @setsid@ man page.
-- For POSIX see, <https://man7.org/linux/man-pages/man2/setsid.2.html setsid>
-- man page.
data Session =
InheritSession -- ^ Inherit the parent session
| NewSession -- ^ Detach process from the current session
Expand All @@ -366,11 +376,15 @@ setSession x (Config cfg) =
NewSession -> cfg { new_session = True}
NewConsole -> cfg {create_new_console = True}

-- | Use the POSIX @setuid@ call to set the user id of the new process before
-- | Use the POSIX
-- <https://man7.org/linux/man-pages/man2/setuid.2.html setuid>
-- call to set the user id of the new process before
-- executing the command. The parent process must have sufficient privileges to
-- set the user id.
--
-- POSIX only. See the POSIX @setuid@ man page.
-- POSIX only. See the POSIX
-- <https://man7.org/linux/man-pages/man2/setuid.2.html setuid>
-- man page.
--
-- Default is 'Nothing' - inherit from the parent.
setUserId :: Maybe Word32 -> Config -> Config
Expand All @@ -382,11 +396,15 @@ setUserId x (Config cfg) =
Config $ cfg { child_user = CUid <$> x }
#endif

-- | Use the POSIX @setgid@ call to set the group id of the new process before
-- | Use the POSIX
-- <https://man7.org/linux/man-pages/man2/setgid.2.html setgid>
-- call to set the group id of the new process before
-- executing the command. The parent process must have sufficient privileges to
-- set the group id.
--
-- POSIX only. See the POSIX @setgid@ man page.
-- POSIX only. See the POSIX
-- <https://man7.org/linux/man-pages/man2/setgid.2.html setgid>
-- man page.
--
-- Default is 'Nothing' - inherit from the parent.
setGroupId :: Maybe Word32 -> Config -> Config
Expand Down Expand Up @@ -414,15 +432,23 @@ setGroupId x (Config cfg) =
-- until the child exits.
--
-- POSIX only. Default is 'False'.
interruptChildOnly :: Bool -> Config -> Config
interruptChildOnly x (Config cfg) = Config $ cfg { delegate_ctlc = x }

{-# DEPRECATED parentIgnoresInterrupt "Use interruptChildOnly instead." #-}
parentIgnoresInterrupt :: Bool -> Config -> Config
parentIgnoresInterrupt x (Config cfg) = Config $ cfg { delegate_ctlc = x }
parentIgnoresInterrupt = interruptChildOnly

-- | On Windows, the parent waits for the entire tree of process i.e. including
-- processes that are spawned by the child process.
-- | On Windows, the parent waits for the entire descendant tree of process
-- i.e. including processes that are spawned by the child process.
--
-- Default is 'True'.
waitForDescendants :: Bool -> Config -> Config
waitForDescendants x (Config cfg) = Config $ cfg { use_process_jobs = x }

{-# DEPRECATED waitForChildTree "Use waitForDescendants instead." #-}
waitForChildTree :: Bool -> Config -> Config
waitForChildTree x (Config cfg) = Config $ cfg { use_process_jobs = x }
waitForChildTree = waitForDescendants

pipeStdErr :: Config -> Config
pipeStdErr (Config cfg) = Config $ cfg { std_err = CreatePipe }
Expand Down Expand Up @@ -585,6 +611,8 @@ pipeChunksWithAction run modCfg path args =

alloc = createProc' modCfg path args

-- | Like 'pipeChunksEither' but use the specified configuration to run the
-- process.
{-# INLINE pipeChunksEitherWith #-}
pipeChunksEitherWith ::
(MonadCatch m, MonadAsync m)
Expand All @@ -604,6 +632,8 @@ pipeChunksEitherWith modifier path args input =
`parallel` fmap Right (toChunksClose stdoutH)
run _ = error "pipeChunksEitherWith: Not reachable"

-- | Like 'pipeChunks' but also includes stderr as 'Left' stream in the
-- 'Either' output.
{-# INLINE pipeChunksEither #-}
pipeChunksEither ::
(MonadCatch m, MonadAsync m)
Expand Down Expand Up @@ -647,6 +677,7 @@ pipeBytesEither path args input =
rightRdr = fmap Right Array.reader
in Stream.unfoldMany (Unfold.either leftRdr rightRdr) output

-- | Like 'pipeChunks' but use the specified configuration to run the process.
{-# INLINE pipeChunksWith #-}
pipeChunksWith ::
(MonadCatch m, MonadAsync m)
Expand Down Expand Up @@ -780,6 +811,8 @@ pipeChars path args input =
-- Generation
-------------------------------------------------------------------------------

-- | Like 'toChunksEither' but use the specified configuration to run the
-- process.
{-# INLINE toChunksEitherWith #-}
toChunksEitherWith ::
(MonadCatch m, MonadAsync m)
Expand All @@ -797,6 +830,7 @@ toChunksEitherWith modifier path args =
`parallel` fmap Right (toChunksClose stdoutH)
run _ = error "toChunksEitherWith: Not reachable"

-- | Like 'toChunks' but use the specified configuration to run the process.
{-# INLINE toChunksWith #-}
toChunksWith ::
(MonadCatch m, MonadAsync m)
Expand Down Expand Up @@ -863,7 +897,7 @@ toBytes path args =
let output = toChunks path args
in Stream.unfoldMany Array.reader output

-- | Like 'toBytes' but generates a stream of @Array Word8@ instead of a stream
-- | Like 'toBytesEither' but generates a stream of @Array Word8@ instead of a stream
-- of @Word8@.
--
-- >>> :{
Expand All @@ -876,7 +910,7 @@ toBytes path args =
--
-- >>> toChunksEither = toChunksEitherWith id
--
-- Prefer 'toChunksEither over 'toBytesEither when performance matters.
-- Prefer 'toChunksEither' over 'toBytesEither' when performance matters.
--
-- /Pre-release/
{-# INLINE toChunksEither #-}
Expand Down Expand Up @@ -970,9 +1004,22 @@ toNull ::
toNull path args = toChunks path args & Stream.fold Fold.drain

-------------------------------------------------------------------------------
-- Process not interacting with the parent process
-- Processes not interacting with the parent process
-------------------------------------------------------------------------------

-- | Launch a standlone process i.e. the process does not have a way to attach
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

standlone -> standalone

-- the IO streams with other processes. The IO streams stdin, stdout, stderr
-- can either be inherited from the parent or closed.
--
-- This API is more powerful than 'interactive' and 'daemon' and can be used to
-- implement both of these. However, it should be used carefully e.g. if you
-- inherit the IO streams and parent is not waiting for the child process to
-- finish then both parent and child may use the IO streams resulting in
-- garbled IO if both are reading/writing simultaneously.
--
-- If the parent chooses to wait for the process an 'ExitCode' is returned
-- otherwise a 'ProcessHandle' is returned which can be used to terminate the
-- process, send signals to it or wait for it to finish.
{-# INLINE standalone #-}
standalone ::
Bool -- ^ Wait for process to finish?
Expand All @@ -998,22 +1045,24 @@ standalone wait (close_stdin, close_stdout, close_stderr) modCfg path args =
s_err = if close_stderr then NoStream else Inherit
in c {std_in = s_in, std_out = s_out, std_err = s_err}

-- | Inherits stdin, stdout, and stderr from the parent, so that the user can
-- interact with the process, user interrupts are handled by the child process,
-- the parent waits for the child process to exit.
-- | Launch a process interfacing with the user. User interrupts are sent to
-- the launched process and ignored by the parent process. The launched process
-- inherits stdin, stdout, and stderr from the parent, so that the user can
-- interact with the process. The parent waits for the child process to exit,
-- an 'ExitCode' is returned when the process finishes.
--
-- This is same as the common @system@ function found in other libraries used
-- to execute commands.
-- This is the same as the common @system@ function found in other libraries
-- used to execute commands.
--
-- On Windows you can pass @setSession NewConsole@ to create a new console.
--
{-# INLINE interactive #-}
interactive ::
{-# INLINE foreground #-}
foreground ::
(Config -> Config)
-> FilePath -- ^ Executable name or path
-> [String] -- ^ Arguments
-> IO ExitCode
interactive modCfg path args =
foreground modCfg path args =
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Can possibly use standalone to implement this.

withCreateProcess cfg (\_ _ _ p -> waitForProcess p)

where
Expand All @@ -1025,11 +1074,24 @@ interactive modCfg path args =
let Config c = modCfg1 $ mkConfig path args
in c {std_in = Inherit, std_out = Inherit, std_err = Inherit}

{-# DEPRECATED interactive "Use foreground instead." #-}
{-# INLINE interactive #-}
interactive ::
(Config -> Config)
-> FilePath -- ^ Executable name or path
-> [String] -- ^ Arguments
-> IO ExitCode
interactive = foreground

-- XXX ProcessHandle can be used to terminate the process. re-export
-- terminateProcess?

-- | Closes stdin, stdout and stderr, creates a new session, detached from the
-- terminal, the parent does not wait for the process to finish.
-- | Launch a daemon process. Closes stdin, stdout and stderr, creates a new
-- session, detached from the terminal, the parent does not wait for the
-- process to finish.
--
-- The 'ProcessHandle' returned can be used to terminate the daemon or send
-- signals to it.
--
{-# INLINE daemon #-}
daemon ::
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Can possibly use standalone to implement this.

Expand Down
38 changes: 31 additions & 7 deletions src/Streamly/System/Process.hs
Original file line number Diff line number Diff line change
Expand Up @@ -35,36 +35,60 @@ module Streamly.System.Process
ProcessFailure (..)

-- * Process Configuration
-- | Use the config modifiers to modify the default config.
, Config

{-
-- ** Common Config Options
-- ** Common Modifiers
-- | These options apply to both POSIX and Windows.
, setCwd
, setEnv
, closeFiles
, newProcessGroup
, Session (..)
, setSession

-- * Posix Only Options
-- ** Posix Only Modifiers
-- | These options have no effect on Windows.
, parentIgnoresInterrupt
, interruptChildOnly
, setUserId
, setGroupId

-- * Windows Only Options
-- ** Windows Only Modifiers
-- | These options have no effect on Posix.
, waitForChildTree
-}
, waitForDescendants

-- * Generation
, toChunks
, toChunksWith
, toBytes
, toChars
, toLines
, toString
, toStdout
, toNull

-- * Transformation
, pipeChunks
, pipeChunksWith
, pipeBytes

-- * Including Stderr Stream
-- | Like other "Generation" routines but along with stdout, stderr is also
-- included in the output stream. stdout is converted to 'Right' values in
-- the output stream and stderr is converted to 'Left' values.
, toBytesEither
, toChunksEither
, toChunksEitherWith
, pipeBytesEither
, pipeChunksEither
, pipeChunksEitherWith

-- * Non-streaming Processes
-- | These processes do not attach the IO streams with other processes.
, foreground
, daemon
, standalone

-- * Deprecated
, processChunks
, processBytes
Expand Down
Loading