mozjs_sys 0.67.1

System crate for the Mozilla SpiderMonkey JavaScript engine.
# This Source Code Form is subject to the terms of the Mozilla Public
# License, v. 2.0. If a copy of the MPL was not distributed with this
# file, You can obtain one at http://mozilla.org/MPL/2.0/.

from __future__ import absolute_import

import os
import platform
import signal
import subprocess
import sys
import time

import mozunit
import pytest

from mozlint.errors import LintersNotConfigured
from mozlint.result import Issue, ResultSummary


here = os.path.abspath(os.path.dirname(__file__))


def test_roll_no_linters_configured(lint, files):
    with pytest.raises(LintersNotConfigured):
        lint.roll(files)


def test_roll_successful(lint, linters, files):
    lint.read(linters('string', 'regex', 'external'))

    result = lint.roll(files)
    assert len(result.issues) == 1
    assert result.failed == set([])

    path = result.issues.keys()[0]
    assert os.path.basename(path) == 'foobar.js'

    errors = result.issues[path]
    assert isinstance(errors, list)
    assert len(errors) == 6

    container = errors[0]
    assert isinstance(container, Issue)
    assert container.rule == 'no-foobar'


def test_roll_from_subdir(lint, linters):
    lint.read(linters('string', 'regex', 'external'))

    oldcwd = os.getcwd()
    try:
        os.chdir(os.path.join(lint.root, 'files'))

        # Path relative to cwd works
        result = lint.roll('no_foobar.js')
        assert len(result.issues) == 0
        assert len(result.failed) == 0
        assert result.returncode == 0

        # Path relative to root doesn't work
        result = lint.roll(os.path.join('files', 'no_foobar.js'))
        assert len(result.issues) == 0
        assert len(result.failed) == 3
        assert result.returncode == 1

        # Paths from vcs are always joined to root instead of cwd
        lint.mock_vcs([os.path.join('files', 'no_foobar.js')])
        result = lint.roll(outgoing=True)
        assert len(result.issues) == 0
        assert len(result.failed) == 0
        assert result.returncode == 0

        result = lint.roll(workdir=True)
        assert len(result.issues) == 0
        assert len(result.failed) == 0
        assert result.returncode == 0
    finally:
        os.chdir(oldcwd)


def test_roll_catch_exception(lint, linters, files, capfd):
    lint.read(linters('raises'))

    lint.roll(files)  # assert not raises
    out, err = capfd.readouterr()
    assert 'LintException' in err


def test_roll_with_global_excluded_path(lint, linters, files):
    lint.exclude = ['**/foobar.js']
    lint.read(linters('string', 'regex', 'external'))
    result = lint.roll(files)

    assert len(result.issues) == 0
    assert result.failed == set([])


def test_roll_with_local_excluded_path(lint, linters, files):
    lint.read(linters('excludes'))
    result = lint.roll(files)

    assert len(result.issues) == 0
    assert result.failed == set([])


def test_roll_with_no_files_to_lint(lint, linters, capfd):
    lint.read(linters('string', 'regex', 'external'))
    lint.mock_vcs([])
    result = lint.roll([], workdir=True)
    assert isinstance(result, ResultSummary)
    assert len(result.issues) == 0
    assert len(result.failed) == 0

    out, err = capfd.readouterr()
    assert 'warning: no files linted' in out


def test_roll_with_invalid_extension(lint, linters, filedir):
    lint.read(linters('external'))
    result = lint.roll(os.path.join(filedir, 'foobar.py'))
    assert len(result.issues) == 0
    assert result.failed == set([])


def test_roll_with_failure_code(lint, linters, files):
    lint.read(linters('badreturncode'))

    result = lint.roll(files, num_procs=1)
    assert len(result.issues) == 0
    assert result.failed == set(['BadReturnCodeLinter'])


def test_roll_warnings(lint, linters, files):
    lint.read(linters('warning'))
    result = lint.roll(files)
    assert len(result.issues) == 0
    assert result.total_issues == 0
    assert len(result.suppressed_warnings) == 1
    assert result.total_suppressed_warnings == 2

    lint.lintargs['show_warnings'] = True
    result = lint.roll(files)
    assert len(result.issues) == 1
    assert result.total_issues == 2
    assert len(result.suppressed_warnings) == 0
    assert result.total_suppressed_warnings == 0


def fake_run_worker(config, paths, **lintargs):
    result = ResultSummary()
    result.issues['count'].append(1)
    return result


@pytest.mark.skipif(platform.system() == 'Windows',
                    reason="monkeypatch issues with multiprocessing on Windows")
@pytest.mark.parametrize('num_procs', [1, 4, 8, 16])
def test_number_of_jobs(monkeypatch, lint, linters, files, num_procs):
    monkeypatch.setattr(sys.modules[lint.__module__], '_run_worker', fake_run_worker)

    linters = linters('string', 'regex', 'external')
    lint.read(linters)
    num_jobs = len(lint.roll(files, num_procs=num_procs).issues['count'])

    if len(files) >= num_procs:
        assert num_jobs == num_procs * len(linters)
    else:
        assert num_jobs == len(files) * len(linters)


@pytest.mark.skipif(platform.system() == 'Windows',
                    reason="monkeypatch issues with multiprocessing on Windows")
@pytest.mark.parametrize('max_paths,expected_jobs', [(1, 12), (4, 6), (16, 6)])
def test_max_paths_per_job(monkeypatch, lint, linters, files, max_paths, expected_jobs):
    monkeypatch.setattr(sys.modules[lint.__module__], '_run_worker', fake_run_worker)

    files = files[:4]
    assert len(files) == 4

    linters = linters('string', 'regex', 'external')[:3]
    assert len(linters) == 3

    lint.MAX_PATHS_PER_JOB = max_paths
    lint.read(linters)
    num_jobs = len(lint.roll(files, num_procs=2).issues['count'])
    assert num_jobs == expected_jobs


@pytest.mark.skipif(platform.system() == 'Windows',
                    reason="signal.CTRL_C_EVENT isn't causing a KeyboardInterrupt on Windows")
def test_keyboard_interrupt():
    # We use two linters so we'll have two jobs. One (string.yml) will complete
    # quickly. The other (slow.yml) will run slowly.  This way the first worker
    # will be be stuck blocking on the ProcessPoolExecutor._call_queue when the
    # signal arrives and the other still be doing work.
    cmd = [sys.executable, 'runcli.py', '-l=string', '-l=slow']
    proc = subprocess.Popen(cmd, stdout=subprocess.PIPE, stderr=subprocess.STDOUT, cwd=here)
    time.sleep(1)
    proc.send_signal(signal.SIGINT)

    out = proc.communicate()[0]
    assert 'warning: not all files were linted' in out
    assert 'Traceback' not in out


def test_support_files(lint, linters, filedir, monkeypatch):
    jobs = []

    # Replace the original _generate_jobs with a new one that simply
    # adds jobs to a list (and then doesn't return anything).
    orig_generate_jobs = lint._generate_jobs

    def fake_generate_jobs(*args, **kwargs):
        jobs.extend([job[1] for job in orig_generate_jobs(*args, **kwargs)])
        return []

    monkeypatch.setattr(lint, '_generate_jobs', fake_generate_jobs)

    linter_path = linters('support_files')[0]
    lint.read(linter_path)

    # Modified support files only lint entire root if --outgoing or --workdir
    # are used.
    path = os.path.join(filedir, 'foobar.js')
    lint.mock_vcs([os.path.join(filedir, 'foobar.py')])
    lint.roll(path)
    assert jobs[0] == [path]

    jobs = []
    lint.roll(path, workdir=True)
    assert jobs[0] == [lint.root]

    jobs = []
    lint.roll(path, outgoing=True)
    assert jobs[0] == [lint.root]

    # Lint config file is implicitly added as a support file
    lint.mock_vcs([linter_path])
    jobs = []
    lint.roll(path, outgoing=True, workdir=True)
    assert jobs[0] == [lint.root]


def test_setup(lint, linters, filedir, capfd):
    with pytest.raises(LintersNotConfigured):
        lint.setup()

    lint.read(linters('setup', 'setupfailed', 'setupraised'))
    lint.setup()
    out, err = capfd.readouterr()
    assert 'setup passed' in out
    assert 'setup failed' in out
    assert 'setup raised' in out
    assert 'error: problem with lint setup, skipping' in out
    assert lint.result.failed_setup == set(['SetupFailedLinter', 'SetupRaisedLinter'])


if __name__ == '__main__':
    mozunit.main()