Viewing File: /home/ubuntu/code_review/arcanist/src/parser/__tests__/ArcanistDiffParserTestCase.php

<?php

/**
 * Test cases for @{class:ArcanistDiffParser}.
 */
final class ArcanistDiffParserTestCase extends PhutilTestCase {

  public function testParser() {
    $root = dirname(__FILE__).'/diff/';
    foreach (Filesystem::listDirectory($root, $hidden = false) as $file) {
      $this->parseDiff($root.$file);
    }
  }

  private function parseDiff($diff_file) {
    $contents = Filesystem::readFile($diff_file);
    $file = basename($diff_file);

    $parser = new ArcanistDiffParser();
    $changes = $parser->parseDiff($contents);

    switch ($file) {
      case 'colorized.hggitdiff':
        $this->assertEqual(1, count($changes));
        break;
      case 'basic-missing-both-newlines-plus.udiff':
      case 'basic-missing-both-newlines.udiff':
      case 'basic-missing-new-newline-plus.udiff':
      case 'basic-missing-new-newline.udiff':
      case 'basic-missing-old-newline-plus.udiff':
      case 'basic-missing-old-newline.udiff':
        $expect_old = strpos($file, '-old-') || strpos($file, '-both-');
        $expect_new = strpos($file, '-new-') || strpos($file, '-both-');
        $expect_two = strpos($file, '-plus');

        $this->assertEqual(count($changes), $expect_two ? 2 : 1);
        $change = reset($changes);
        $this->assertTrue($change !== null);

        $hunks = $change->getHunks();
        $this->assertEqual(1, count($hunks));

        $hunk = reset($hunks);
        $this->assertEqual((bool)$expect_old, $hunk->getIsMissingOldNewline());
        $this->assertEqual((bool)$expect_new, $hunk->getIsMissingNewNewline());
        break;
      case 'basic-binary.udiff':
        $this->assertEqual(1, count($changes));
        $change = reset($changes);
        $this->assertEqual(
          ArcanistDiffChangeType::FILE_BINARY,
          $change->getFileType());
        break;
      case 'basic-multi-hunk.udiff':
        $this->assertEqual(1, count($changes));
        $change = reset($changes);
        $hunks = $change->getHunks();
        $this->assertEqual(4, count($hunks));
        $this->assertEqual('right', $change->getCurrentPath());
        $this->assertEqual('left', $change->getOldPath());
        break;
      case 'basic-multi-hunk-content.svndiff':
        $this->assertEqual(1, count($changes));
        $change = reset($changes);
        $hunks = $change->getHunks();
        $this->assertEqual(2, count($hunks));

        $there_is_a_literal_trailing_space_here = ' ';

        $corpus_0 = <<<EOCORPUS
 asdfasdf
+% quack
 %
-%
 %%
 %%
 %%%

EOCORPUS;
        $corpus_1 = <<<EOCORPUS
 %%%%%
 %%%%%
{$there_is_a_literal_trailing_space_here}
-!
+! quack

EOCORPUS;
        $this->assertEqual(
          $corpus_0,
          $hunks[0]->getCorpus());
        $this->assertEqual(
          $corpus_1,
          $hunks[1]->getCorpus());
        break;
      case 'svn-ignore-whitespace-only.svndiff':
        $this->assertEqual(2, count($changes));
        $hunks = reset($changes)->getHunks();
        $this->assertEqual(0, count($hunks));
        break;
      case 'svn-property-add.svndiff':
        $this->assertEqual(1, count($changes));
        $change = reset($changes);
        $hunks = reset($changes)->getHunks();
        $this->assertEqual(1, count($hunks));
        $this->assertEqual(
          array(
            'duck' => 'quack',
          ),
          $change->getNewProperties());
        break;
      case 'svn-property-modify.svndiff':
        $this->assertEqual(2, count($changes));

        $change = array_shift($changes);
        $this->assertEqual(0, count($change->getHunks()));
        $this->assertEqual(
          array(
            'svn:ignore' => '*.phpz',
          ),
          $change->getOldProperties());
        $this->assertEqual(
          array(
            'svn:ignore' => '*.php',
          ),
          $change->getNewProperties());

        $change = array_shift($changes);
        $this->assertEqual(0, count($change->getHunks()));
        $this->assertEqual(
          array(
            'svn:special' => '*',
          ),
          $change->getOldProperties());
        $this->assertEqual(
          array(
            'svn:special' => 'moo',
          ),
          $change->getNewProperties());
        break;
      case 'svn-property-delete.svndiff':
        $this->assertEqual(1, count($changes));
        $change = reset($changes);

        $this->assertEqual(0, count($change->getHunks()));
        $this->assertEqual(
          $change->getOldProperties(),
          array(
            'svn:special' => '*',
          ));
        $this->assertEqual(
          array(
          ),
          $change->getNewProperties());
        break;
      case 'svn-property-merged.svndiff':
        $this->assertEqual(1, count($changes));
        $change = reset($changes);

        $this->assertEqual(count($change->getHunks()), 0);

        $this->assertEqual(
          $change->getOldProperties(),
          array());
        $this->assertEqual(
          $change->getNewProperties(),
          array());
        break;
      case 'svn-property-merge.svndiff':
        $this->assertEqual(1, count($changes));
        $change = reset($changes);

        $this->assertEqual(count($change->getHunks()), 0);
        $this->assertEqual(
          $change->getOldProperties(),
          array(
          ));
        $this->assertEqual(
          $change->getNewProperties(),
          array(
            'svn:mergeinfo' => <<<EOTEXT
Merged /tfb/branches/internmove/www/html/js/help/UIFaq.js:r83462-126155
Merged /tfb/branches/ads-create-v3/www/html/js/help/UIFaq.js:r140558-142418
EOTEXT
          ));
        break;
      case 'svn-property-older-than-1.5.svndiff':
        // In SVN 1.5, the format for property diffs changed to use the words
        // "Added", "Deleted" and "Modified" instead of "Name". This is an old
        // property change diff which uses "Name".
        $this->assertEqual(1, count($changes));
        $change = reset($changes);

        $this->assertEqual(count($change->getHunks()), 0);
        $this->assertEqual(
          $change->getOldProperties(),
          array(
          ));
        $this->assertEqual(
          $change->getNewProperties(),
          array(
            'svn:executable' => '*',
          ));
        break;
      case 'svn-binary-add.svndiff':
        $this->assertEqual(1, count($changes));
        $change = reset($changes);
        $this->assertEqual(
          ArcanistDiffChangeType::FILE_BINARY,
          $change->getFileType());
        $this->assertEqual(0, count($change->getHunks()));
        $this->assertEqual(
          array(
            'svn:mime-type' => 'application/octet-stream',
          ),
          $change->getNewProperties());
        break;
      case 'svn-binary-diff.svndiff':
      case 'svn-binary-diff-freebsd.svndiff':
        $this->assertEqual(1, count($changes));
        $change = reset($changes);
        $this->assertEqual(
          ArcanistDiffChangeType::FILE_BINARY,
          $change->getFileType());
        $this->assertEqual(count($change->getHunks()), 0);
        break;
      case 'git-delete-file.gitdiff':
        $this->assertEqual(1, count($changes));
        $change = reset($changes);
        $this->assertEqual(
          ArcanistDiffChangeType::TYPE_DELETE,
          $change->getType());
        $this->assertEqual(
          'scripts/intern/test/testfile2',
          $change->getCurrentPath());
        $this->assertEqual(1, count($change->getHunks()));
        break;
      case 'git-binary-change.gitdiff':
        $this->assertEqual(1, count($changes));
        $change = reset($changes);
        $this->assertEqual(
          ArcanistDiffChangeType::FILE_BINARY,
          $change->getFileType());
        $this->assertEqual(0, count($change->getHunks()));
        break;
      case 'git-filemode-change.gitdiff':
        $this->assertEqual(1, count($changes));
        $change = reset($changes);
        $this->assertEqual(1, count($change->getHunks()));
        $this->assertEqual(
          array(
            'unix:filemode' => '100644',
          ),
          $change->getOldProperties());
        $this->assertEqual(
          array(
            'unix:filemode' => '100755',
          ),
          $change->getNewProperties());
        break;
      case 'git-filemode-change-only.gitdiff':
        $this->assertEqual(count($changes), 2);
        $change = reset($changes);
        $this->assertEqual(count($change->getHunks()), 0);
        $this->assertEqual(
          array(
            'unix:filemode' => '100644',
          ),
          $change->getOldProperties());
        $this->assertEqual(
          array(
            'unix:filemode' => '100755',
          ),
          $change->getNewProperties());
        break;
      case 'svn-empty-file.svndiff':
        $this->assertEqual(2, count($changes));
        $change = array_shift($changes);
        $this->assertEqual(0, count($change->getHunks()));
        break;
      case 'git-ignore-whitespace-only.gitdiff':
        $this->assertEqual(count($changes), 2);

        $change = array_shift($changes);
        $this->assertEqual(count($change->getHunks()), 0);
        $this->assertEqual(
          $change->getOldPath(),
          'scripts/intern/test/testfile2');
        $this->assertEqual(
          $change->getCurrentPath(),
          'scripts/intern/test/testfile2');

        $change = array_shift($changes);
        $this->assertEqual(count($change->getHunks()), 1);
        $this->assertEqual(
          $change->getOldPath(),
          'scripts/intern/test/testfile3');
        $this->assertEqual(
          $change->getCurrentPath(),
          'scripts/intern/test/testfile3');
        break;
      case 'git-move.gitdiff':
      case 'git-move-edit.gitdiff':
      case 'git-move-plus.gitdiff':

        $extra_changeset = (bool)strpos($file, '-plus');
        $has_hunk = (bool)strpos($file, '-edit');

        $this->assertEqual($extra_changeset ? 3 : 2, count($changes));

        $change = array_shift($changes);
        $this->assertEqual($has_hunk ? 1 : 0,
                          count($change->getHunks()));
        $this->assertEqual(
          $change->getType(),
          ArcanistDiffChangeType::TYPE_MOVE_HERE);

        $target = $change;

        $change = array_shift($changes);
        $this->assertEqual(0, count($change->getHunks()));
        $this->assertEqual(
          ArcanistDiffChangeType::TYPE_MOVE_AWAY,
          $change->getType());

        $this->assertEqual(
          $change->getCurrentPath(),
          $target->getOldPath());
        $this->assertTrue(
          in_array($target->getCurrentPath(), $change->getAwayPaths()));
        break;
      case 'git-merge-header.gitdiff':
        $this->assertEqual(1, count($changes));
        $change = reset($changes);
        $this->assertEqual(
          ArcanistDiffChangeType::TYPE_MESSAGE,
          $change->getType());
        $this->assertEqual(
          '501f6d519703458471dbea6284ec5f49d1408598',
          $change->getCommitHash());
        break;
      case 'git-new-file.gitdiff':
        $this->assertEqual(1, count($changes));
        $change = reset($changes);
        $this->assertEqual(
          ArcanistDiffChangeType::TYPE_ADD,
          $change->getType());
        break;
      case 'git-copy.gitdiff':
        $this->assertEqual(2, count($changes));

        $change = array_shift($changes);
        $this->assertEqual(0, count($change->getHunks()));
        $this->assertEqual(
          ArcanistDiffChangeType::TYPE_COPY_HERE,
          $change->getType());
        $this->assertEqual(
          'flib/intern/widgets/ui/UIWidgetRSSBox.php',
          $change->getCurrentPath());

        $change = array_shift($changes);
        $this->assertEqual(0, count($change->getHunks()));
        $this->assertEqual(
          ArcanistDiffChangeType::TYPE_COPY_AWAY,
          $change->getType());
        $this->assertEqual(
          'lib/display/intern/ui/widget/UIWidgetRSSBox.php',
          $change->getCurrentPath());

        break;
      case 'git-copy-plus.gitdiff':
        $this->assertEqual(2, count($changes));

        $change = array_shift($changes);
        $this->assertEqual(3, count($change->getHunks()));
        $this->assertEqual(
          ArcanistDiffChangeType::TYPE_COPY_HERE,
          $change->getType());
        $this->assertEqual(
          'flib/intern/widgets/ui/UIWidgetGraphConnect.php',
          $change->getCurrentPath());

        $change = array_shift($changes);
        $this->assertEqual(0, count($change->getHunks()));
        $this->assertEqual(
          ArcanistDiffChangeType::TYPE_COPY_AWAY,
          $change->getType());
        $this->assertEqual(
          'lib/display/intern/ui/widget/UIWidgetLunchtime.php',
          $change->getCurrentPath());
        break;
      case 'svn-property-multiline.svndiff':
        $this->assertEqual(1, count($changes));
        $change = array_shift($changes);

        $this->assertEqual(0, count($change->getHunks()));
        $this->assertEqual(
          array(
            'svn:ignore' => 'tags',
          ),
          $change->getOldProperties());
        $this->assertEqual(
          array(
            'svn:ignore' => "tags\nasdf\nlol\nwhat",
          ),
          $change->getNewProperties());
        break;
      case 'git-empty-files.gitdiff':
        $this->assertEqual(2, count($changes));
        while ($change = array_shift($changes)) {
          $this->assertEqual(0, count($change->getHunks()));
        }
        break;
      case 'git-mnemonicprefix.gitdiff':
        // Check parsing of diffs created with `diff.mnemonicprefix`
        // configuration option set to `true`.
        $this->assertEqual(1, count($changes));
        $this->assertEqual(1, count(reset($changes)->getHunks()));
        break;
      case 'git-commit.gitdiff':
      case 'git-commit-logdecorate.gitdiff':
        $this->assertEqual(1, count($changes));
        $change = reset($changes);
        $this->assertEqual(
          ArcanistDiffChangeType::TYPE_MESSAGE,
          $change->getType());
        $this->assertEqual(
          '76e2f1339c298c748aa0b52030799ed202a6537b',
          $change->getCommitHash());
        $this->assertEqual(
          <<<EOTEXT

Deprecating UIActionButton (Part 1)

Summary: Replaces calls to UIActionButton with <ui:button>.  I tested most
         of these calls, but there were some that I didn't know how to
         reach, so if you are one of the owners of this code, please test
         your feature in my sandbox: www.ngao.devrs013.facebook.com

         @brosenthal, I removed some logic that was setting a disabled state
         on a UIActionButton, which is actually a no-op.

Reviewed By: brosenthal

Other Commenters: sparker, egiovanola

Test Plan: www.ngao.devrs013.facebook.com

           Explicitly tested:
           * ads creation flow (add keyword)
           * ads manager (conversion tracking)
           * help center (create a discussion)
           * new user wizard (next step button)

Revert: OK

DiffCamp Revision: 94064

git-svn-id: svn+ssh://tubbs/svnroot/tfb/trunk/www@223593 2c7ba8d8
EOTEXT
          , $change->getMetadata('message'));
        break;
      case 'git-binary.gitdiff':
        $this->assertEqual(1, count($changes));
        $change = reset($changes);
        $this->assertEqual(
          ArcanistDiffChangeType::TYPE_CHANGE,
          $change->getType());
        $this->assertEqual(
          ArcanistDiffChangeType::FILE_BINARY,
          $change->getFileType());
        break;
      case 'git-odd-filename.gitdiff':
        $this->assertEqual(2, count($changes));
        $change = reset($changes);
        $this->assertEqual(
          'old/'."\342\210\206".'.jpg',
          $change->getOldPath());
        $this->assertEqual(
          'new/'."\342\210\206".'.jpg',
          $change->getCurrentPath());
        break;
      case 'hg-binary-change.hgdiff':
      case 'hg-solo-binary-change.hgdiff':
        $this->assertEqual(1, count($changes));
        $change = reset($changes);
        $this->assertEqual(
          ArcanistDiffChangeType::TYPE_ADD,
          $change->getType());
        $this->assertEqual(
          ArcanistDiffChangeType::FILE_BINARY,
          $change->getFileType());
        break;
      case 'hg-binary-delete.hgdiff':
        $this->assertEqual(1, count($changes));
        $change = reset($changes);
        $this->assertEqual(
          ArcanistDiffChangeType::TYPE_DELETE,
          $change->getType());
        $this->assertEqual(
          ArcanistDiffChangeType::FILE_BINARY,
          $change->getFileType());
        break;
      case 'git-replace-symlink.gitdiff':
        $this->assertEqual(1, count($changes));
        $change = array_shift($changes);
        $this->assertEqual(
          ArcanistDiffChangeType::TYPE_CHANGE,
          $change->getType());
        break;
      case 'svn-1.7-property-added.svndiff':
        $this->assertEqual(1, count($changes));
        $change = head($changes);
        $new_properties = $change->getNewProperties();
        $this->assertEqual(2, count($new_properties));
        $this->assertEqual('*', idx($new_properties, 'svn:executable'));
        $this->assertEqual('text/html', idx($new_properties, 'svn:mime-type'));
        break;
      case 'hg-diff-range.hgdiff':
        $this->assertEqual(1, count($changes));
        $change = array_shift($changes);
        $this->assertEqual(
          'Test.java',
          $change->getOldPath());
        $this->assertEqual(
          'Test.java',
          $change->getCurrentPath());
        break;
      case 'hg-patch.hgdiff':
        $this->assertEqual(1, count($changes));
        break;
      case 'hg-patch-git.hgdiff':
        $this->assertEqual(1, count($changes));
        break;
      case 'custom-prefixes.gitdiff':
        $this->assertEqual(1, count($changes));
        $change = head($changes);
        $this->assertEqual(
          'file',
          $change->getCurrentPath());
        break;
      case 'custom-prefixes-edit.gitdiff':
        $this->assertEqual(1, count($changes));
        $change = head($changes);
        $this->assertEqual(
          'file',
          $change->getCurrentPath());
        break;
      case 'more-newlines.svndiff':
        $this->assertEqual(1, count($changes));
        break;
      case 'suppress-blank-empty.gitdiff':
        $this->assertEqual(1, count($changes));
        break;
      case 'svn-property-windows.svndiff':
        $this->assertEqual(1, count($changes));
        break;
      case 'rcs-addline.rcsdiff':
        $this->assertEqual(1, count($changes));
        $change = array_shift($changes);
        $this->assertEqual(
          ArcanistDiffChangeType::TYPE_CHANGE,
          $change->getType());
        break;
      case 'rcs-deleteline.rcsdiff':
        $this->assertEqual(1, count($changes));
        $change = array_shift($changes);
        $this->assertEqual(
          ArcanistDiffChangeType::TYPE_CHANGE,
          $change->getType());
        break;
      case 'comment.svndiff':
        $this->assertEqual(1, count($changes));
        $change = array_shift($changes);
        $this->assertEqual(
          ArcanistDiffChangeType::TYPE_CHANGE,
          $change->getType());
        break;
      case 'svnlook-basics.svndiff':
      case 'svnlook-add.svndiff':
      case 'svnlook-delete.svndiff':
      case 'svnlook-copied.svndiff':
        $this->assertEqual(1, count($changes));
        break;
      case 'git-format-patch.gitdiff':
        $this->assertEqual(2, count($changes));

        $change = array_shift($changes);
        $this->assertEqual(
          ArcanistDiffChangeType::TYPE_MESSAGE,
          $change->getType());
        $this->assertEqual('WIP', $change->getMetadata('message'));

        $change = array_shift($changes);
        $this->assertEqual(
          ArcanistDiffChangeType::TYPE_CHANGE,
          $change->getType());
        break;
      case 'svn-double-diff.svndiff':
        $this->assertEqual(1, count($changes));

        $change = array_shift($changes);
        $hunks = $change->getHunks();
        $this->assertEqual(1, count($hunks));
        break;
      case 'git-remove-spaces.gitdiff':
        $this->assertEqual(1, count($changes));

        $change = array_shift($changes);
        $this->assertEqual('file with spaces.txt', $change->getOldPath());
        break;
      default:
        throw new Exception(pht('No test block for diff file %s.', $diff_file));
        break;
    }
  }

  public function testGitCommonFilenameExtraction() {
    static $tests = array(
      'a/filename.c b/filename.c'         => 'filename.c',
      "a/filename.c b/filename.c\n"       => 'filename.c',
      "a/filename.c b/filename.c\r\n"     => 'filename.c',
      'filename.c filename.c'             => 'filename.c',
      '1/filename.c 2/filename.c'         => 'filename.c',
      '"a/\\"quotes\\"" "b/\\"quotes\\""' => '"quotes"',
      '"a/\\"quotes and spaces\\"" "b/\\"quotes and spaces\\""' =>
        '"quotes and spaces"',
      '"a/\\342\\230\\203" "b/\\342\\230\\203"' =>
         "\xE2\x98\x83",
      'a/Core Data/filename.c b/Core Data/filename.c' =>
         'Core Data/filename.c',
      'some file with spaces.c some file with spaces.c' =>
         'some file with spaces.c',
      '"foo bar.c" foo bar.c'        => 'foo bar.c',
      '"a/foo bar.c" b/foo bar.c'    => 'foo bar.c',
      'src/file dst/file'            => 'file',

      // Renames are handled by the "rename from ..." lines later in
      // the diff, for simplicity of parsing; this is also how git
      // itself handles it.
      'a/foo.c b/bar.c'              => null,
      'a/foo bar.c b/baz troz.c'     => null,
      '"a/foo bar.c" b/baz troz.c'   => null,
      'a/foo bar.c "b/baz troz.c"'   => null,
      '"a/foo bar.c" "b/baz troz.c"' => null,
      'filename file with spaces.c filename file with spaces.c' =>
        'filename file with spaces.c',
    );

    foreach ($tests as $input => $expect) {
      $result = ArcanistDiffParser::extractGitCommonFilename($input);
      $this->assertEqual(
        $expect,
        $result,
        pht('Split: %s', $input));
    }
  }


  public function runSingleRename($diffline, $from, $to, $old, $new) {
    $str = "diff --git $diffline\nsimilarity index 95%\n"
         ."rename from $from\nrename to $to\n";
    $parser = new ArcanistDiffParser();
    $changes = $parser->parseDiff($str);
    $this->assertTrue(
      $changes !== null,
      pht("Parsed:\n%s", $str));
    $this->assertEqual(
      $old == $new ? 1 : 2, count($changes),
      pht("Parsed one change:\n%s", $str));
    $change = reset($changes);
    $this->assertEqual(
      array($old, $new),
      array($change->getOldPath(), $change->getCurrentPath()),
      pht('Split: %s', $diffline));
  }

  public function testGitRenames() {
    $this->runSingleRename('a/old.c b/new.c',
                           'old.c',                   'new.c',
                           'old.c',                   'new.c');
    $this->runSingleRename('old.c new.c',
                           'old.c',                   'new.c',
                           'old.c',                   'new.c');
    $this->runSingleRename('1/old.c 2/new.c',
                           'old.c',                   'new.c',
                           'old.c',                   'new.c');
    $this->runSingleRename('from/file.c to/file.c',
                           'from/file.c',             'to/file.c',
                           'from/file.c',             'to/file.c');
    $this->runSingleRename('"a/\\"quotes1\\"" "b/\\"quotes2\\""',
                           '"\\"quotes1\\""',         '"\\"quotes2\\""',
                           '"quotes1"',               '"quotes2"');
    $this->runSingleRename('"a/\\"quotes spaces1\\"" "b/\\"quotes spaces2\\""',
                           '"\\"quotes spaces1\\""',  '"\\"quotes spaces2\\""',
                           '"quotes spaces1"',        '"quotes spaces2"');
    $this->runSingleRename('"a/\\342\\230\\2031" "b/\\342\\230\\2032"',
                           '"\\342\\230\\2031"',      '"\\342\\230\\2032"',
                           "\xE2\x98\x831",           "\xE2\x98\x832");
    $this->runSingleRename('a/Core Data/old.c b/Core Data/new.c',
                           'Core Data/old.c',         'Core Data/new.c',
                           'Core Data/old.c',         'Core Data/new.c');
    $this->runSingleRename('file with spaces.c file with spaces.c',
                           'file with spaces.c', 'file with spaces.c',
                           'file with spaces.c', 'file with spaces.c');
    $this->runSingleRename('a/non-quoted filename.c "b/quoted filename.c"',
                           'non-quoted filename.c',   '"quoted filename.c"',
                           'non-quoted filename.c',   'quoted filename.c');
    $this->runSingleRename('non-quoted filename.c "quoted filename.c"',
                           'non-quoted filename.c',   '"quoted filename.c"',
                           'non-quoted filename.c',   'quoted filename.c');
    $this->runSingleRename('"a/quoted filename.c" b/non quoted filename.c',
                           '"quoted filename.c"',     'non quoted filename.c',
                           'quoted filename.c',       'non quoted filename.c');
    $this->runSingleRename('"quoted filename.c" non-quoted filename.c',
                           '"quoted filename.c"',     'non-quoted filename.c',
                           'quoted filename.c',       'non-quoted filename.c');
    $this->runSingleRename('old file with spaces.c new file with spaces.c',
                           'old file with spaces.c',  'new file with spaces.c',
                           'old file with spaces.c',  'new file with spaces.c');
    $this->runSingleRename('old file old file',
                           'old file old',            'file',
                           'old file old',            'file');
  }
}
Back to Directory File Manager