Source code for mythx_cli.formatter.simple_stdout

"""This module contains a simple text formatter class printing a subset of the
response data."""

from typing import List, Optional, Tuple

from mythx_models.response import (
    AnalysisInputResponse,
    AnalysisListResponse,
    AnalysisStatusResponse,
    DetectedIssuesResponse,
    GroupListResponse,
    GroupStatusResponse,
    VersionResponse,
)

from .base import BaseFormatter
from .util import generate_dashboard_link, get_source_location_by_offset


[docs]class SimpleFormatter(BaseFormatter): """The simple text formatter. This formatter generates simplified text output. It also displays the source locations of issues by line in the Solidity source code if given. Therefore, this formatter requires the analysis input to be given. """ report_requires_input = True
[docs] @staticmethod def format_analysis_list(resp: AnalysisListResponse) -> str: """Format an analysis list response to a simple text representation.""" res = [] for analysis in resp: res.append("UUID: {}".format(analysis.uuid)) res.append("Submitted at: {}".format(analysis.submitted_at)) res.append("Status: {}".format(analysis.status)) res.append("") return "\n".join(res)
[docs] @staticmethod def format_group_status(resp: GroupStatusResponse) -> str: """Format a group status response to a simple text representation.""" res = [ "ID: {}".format(resp.group.identifier), "Name: {}".format(resp.group.name or "<unnamed>"), "Created on: {}".format(resp.group.created_at), "Status: {}".format(resp.group.status), "", ] return "\n".join(res)
[docs] @staticmethod def format_group_list(resp: GroupListResponse) -> str: """Format an analysis group response to a simple text representation.""" res = [] for group in resp: res.append("ID: {}".format(group.identifier)) res.append("Name: {}".format(group.name or "<unnamed>")) res.append("Created on: {}".format(group.created_at)) res.append("Status: {}".format(group.status)) res.append("") return "\n".join(res)
[docs] @staticmethod def format_analysis_status(resp: AnalysisStatusResponse) -> str: """Format an analysis status response to a simple text representation.""" res = [ "UUID: {}".format(resp.uuid), "Submitted at: {}".format(resp.submitted_at), "Status: {}".format(resp.status), "", ] return "\n".join(res)
[docs] @staticmethod def format_detected_issues( issues_list: List[ Tuple[DetectedIssuesResponse, Optional[AnalysisInputResponse]] ] ) -> str: """Format an issue report to a simple text representation.""" # TODO: Sort by file for resp, inp in issues_list: res = [] for report in resp.issue_reports: for issue in report.issues: res.append(generate_dashboard_link(resp.uuid)) res.append( "Title: {} ({})".format(issue.swc_title or "-", issue.severity) ) res.append("Description: {}".format(issue.description_long.strip())) for loc in issue.locations: comp = loc.source_map.components[0] source_list = loc.source_list or report.source_list if source_list and 0 >= comp.file_id < len(source_list): filename = source_list[comp.file_id] if not inp.sources or filename not in inp.sources: # Skip files we don't have source for # (e.g. with unresolved bytecode hashes) res.append("") continue line = get_source_location_by_offset( inp.sources[filename]["source"], comp.offset ) snippet = inp.sources[filename]["source"].split("\n")[ line - 1 ] res.append("{}:{}".format(filename, line)) res.append("\t" + snippet.strip()) res.append("") return "\n".join(res)
[docs] @staticmethod def format_version(resp: VersionResponse) -> str: """Format a version response to a simple text representation.""" return "\n".join( [ "API: {}".format(resp.api_version), "Harvey: {}".format(resp.harvey_version), "Maru: {}".format(resp.maru_version), "Mythril: {}".format(resp.mythril_version), "Hashed: {}".format(resp.hashed_version), ] )