MediaWiki REL1_31
CommandTest.php
Go to the documentation of this file.
1<?php
2
5use Wikimedia\TestingAccessWrapper;
6
11class CommandTest extends PHPUnit\Framework\TestCase {
12
13 use MediaWikiCoversValidator;
14
15 private function requirePosix() {
16 if ( wfIsWindows() ) {
17 $this->markTestSkipped( 'This test requires a POSIX environment.' );
18 }
19 }
20
24 public function testExecute( $command, $args, $expectedExitCode, $expectedOutput ) {
26 $result = $command
27 ->params( $args )
28 ->execute();
29
30 $this->assertSame( $expectedExitCode, $result->getExitCode() );
31 $this->assertSame( $expectedOutput, $result->getStdout() );
32 }
33
34 public function provideExecute() {
35 return [
36 'success status' => [ 'success_status.php', [], 0, '' ],
37 'failure status' => [ 'failure_status.php', [], 1, '' ],
38 'output' => [ 'echo_args.php', [ 'x', '>', 'y' ], 0, 'x > y' ],
39 ];
40 }
41
42 public function testEnvironment() {
43 $command = $this->getPhpCommand( 'echo_env.php' );
44 $result = $command
45 ->params( [ 'FOO' ] )
46 ->environment( [ 'FOO' => 'bar' ] )
47 ->execute();
48 $this->assertSame( "bar", $result->getStdout() );
49 }
50
51 public function testStdout() {
52 $command = $this->getPhpCommand( 'echo_args.php' );
53
54 $result = $command
55 ->unsafeParams( 'ThisIsStderr', '1>&2' )
56 ->execute();
57
58 $this->assertNotContains( 'ThisIsStderr', $result->getStdout() );
59 $this->assertEquals( "ThisIsStderr", $result->getStderr() );
60 }
61
62 public function testStdoutRedirection() {
63 // The double redirection doesn't work on Windows
64 $this->requirePosix();
65
66 $command = new Command();
67
68 $result = $command
69 ->params( 'bash', '-c', 'echo ThisIsStderr 1>&2' )
70 ->includeStderr( true )
71 ->execute();
72
73 $this->assertEquals( "ThisIsStderr\n", $result->getStdout() );
74 $this->assertNull( $result->getStderr() );
75 }
76
77 public function testOutput() {
78 $command = $this->getPhpCommand(
79 'stdout_stderr.php',
80 [ 'correct stdout' ]
81 );
82 $result = $command->execute();
83 $this->assertSame( 'correct stdout', $result->getStdout() );
84 $this->assertSame( null, $result->getStderr() );
85
86 $command = $this->getPhpCommand(
87 'stdout_stderr.php',
88 [ 'correct stdout ', 'correct stderr ' ]
89 );
90 $result = $command
91 ->includeStderr()
92 ->execute();
93 $this->assertRegExp( '/correct stdout/m', $result->getStdout() );
94 $this->assertRegExp( '/correct stderr/m', $result->getStdout() );
95 $this->assertSame( null, $result->getStderr() );
96
97 $command = $this->getPhpCommand(
98 'stdout_stderr.php',
99 [ 'correct stdout', 'correct stderr' ]
100 );
101 $result = $command
102 ->execute();
103 $this->assertSame( 'correct stdout', $result->getStdout() );
104 $this->assertSame( 'correct stderr', $result->getStderr() );
105 }
106
110 public function testNullsAreSkipped() {
111 $command = TestingAccessWrapper::newFromObject( new Command );
112 $command->params( 'echo', 'a', null, 'b' );
113 $command->unsafeParams( 'c', null, 'd' );
114
115 if ( wfIsWindows() ) {
116 $this->assertEquals( '"echo" "a" "b" c d', $command->command );
117 } else {
118 $this->assertEquals( "'echo' 'a' 'b' c d", $command->command );
119 }
120 }
121
122 public function testT69870() {
123 // Testing for Bug T69870
124 // wfShellExec() cuts off stdout at multiples of 8192 bytes.
125
126 // hangs on Windows, see Bug T199989, non-blocking pipes
127 $this->requirePosix();
128
129 // Test several times because it involves a race condition that may randomly succeed or fail
130 for ( $i = 0; $i < 10; $i++ ) {
131 $command = $this->getPhpCommand( 'echo_333333_stars.php' );
133 ->execute()
134 ->getStdout();
135 $this->assertEquals( 333333, strlen( $output ) );
136 }
137 }
138
139 public function testLogStderr() {
140 $logger = new TestLogger( true, function ( $message, $level, $context ) {
141 return $level === Psr\Log\LogLevel::ERROR ? '1' : null;
142 }, true );
143 $command = $this->getPhpCommand( 'echo_args.php' );
144 $command->setLogger( $logger );
145 $command->unsafeParams( 'ThisIsStderr', '1>&2' );
146 $command->execute();
147 $this->assertEmpty( $logger->getBuffer() );
148
149 $command = $this->getPhpCommand( 'echo_args.php' );
150 $command->setLogger( $logger );
151 $command->logStderr();
152 $command->unsafeParams( 'ThisIsStderr', '1>&2' );
153 $command->execute();
154 $this->assertSame( 1, count( $logger->getBuffer() ) );
155 $this->assertSame( trim( $logger->getBuffer()[0][2]['error'] ), 'ThisIsStderr' );
156 }
157
158 public function testInput() {
159 // hangs on Windows, see Bug T199989, non-blocking pipes
160 $this->requirePosix();
161
162 $command = $this->getPhpCommand( 'echo_stdin.php' );
163 $command->input( 'abc' );
164 $result = $command->execute();
165 $this->assertSame( 'abc', $result->getStdout() );
166
167 // now try it with something that does not fit into a single block
168 $command = $this->getPhpCommand( 'echo_stdin.php' );
169 $command->input( str_repeat( '!', 1000000 ) );
170 $result = $command->execute();
171 $this->assertSame( 1000000, strlen( $result->getStdout() ) );
172
173 // And try it with empty input
174 $command = $this->getPhpCommand( 'echo_stdin.php' );
175 $command->input( '' );
176 $result = $command->execute();
177 $this->assertSame( '', $result->getStdout() );
178 }
179
184 public function testDisablingRestrictions() {
185 $command = TestingAccessWrapper::newFromObject( new Command() );
186 // As CommandFactory does for the firejail case:
187 $command->restrict( Shell::RESTRICT_DEFAULT );
188 // Disable restrictions
189 $command->restrict( Shell::RESTRICT_NONE );
190 $this->assertSame( 0, $command->restrictions );
191 }
192
205 private function getPhpCommand( $fileName, array $args = [] ) {
206 $command = new Command;
207 $params = [
208 PHP_BINARY,
209 __DIR__
210 . DIRECTORY_SEPARATOR
211 . 'bin'
212 . DIRECTORY_SEPARATOR
213 . $fileName
214 ];
215 $params = array_merge( $params, $args );
216
217 $command->params( $params );
218 $command->limits( [ 'memory' => 0 ] );
219 return $command;
220 }
221}
wfIsWindows()
Check if the operating system is Windows.
$command
Definition cdb.php:65
if( $line===false) $args
Definition cdb.php:64
\MediaWiki\Shell\Command Shell
testDisablingRestrictions()
Ensure that it's possible to disable the default shell restrictions.
getPhpCommand( $fileName, array $args=[])
Creates a command that will execute one of the PHP test scripts by its file name, using the current P...
testNullsAreSkipped()
Test that null values are skipped by params() and unsafeParams()
testExecute( $command, $args, $expectedExitCode, $expectedOutput)
provideExecute
testStdoutRedirection()
Class used for executing shell commands.
Definition Command.php:35
Executes shell commands.
Definition Shell.php:44
A logger that may be configured to either buffer logs or to print them to the output where PHPUnit wi...
static configuration should be added through ResourceLoaderGetConfigVars instead can be used to get the real title after the basic globals have been set but before ordinary actions take place $output
Definition hooks.txt:2255
do that in ParserLimitReportFormat instead use this to modify the parameters of the image all existing parser cache entries will be invalid To avoid you ll need to handle that somehow(e.g. with the RejectParserCacheValue hook) because MediaWiki won 't do it for you. & $defaults also a ContextSource after deleting those rows but within the same transaction you ll probably need to make sure the header is varied on and they can depend only on the ResourceLoaderContext $context
Definition hooks.txt:2811
null means default in associative array with keys and values unescaped Should be merged with default with a value of false meaning to suppress the attribute in associative array with keys and values unescaped noclasses just before the function returns a value If you return true
Definition hooks.txt:2006
$params