mirror of
https://github.com/denoland/deno.git
synced 2024-12-30 11:16:38 -05:00
cca8856fbf
This removes the dependency on global state and instead relies on the runtime's internal state to get the script sources it saw when it collected code coverage for them.
368 lines
8.9 KiB
Rust
368 lines
8.9 KiB
Rust
// Copyright 2018-2020 the Deno authors. All rights reserved. MIT license.
|
|
|
|
use crate::colors;
|
|
use crate::inspector::DenoInspector;
|
|
use deno_core::error::generic_error;
|
|
use deno_core::error::AnyError;
|
|
use deno_core::futures::channel::oneshot;
|
|
use deno_core::serde_json;
|
|
use deno_core::serde_json::json;
|
|
use deno_core::url::Url;
|
|
use deno_core::v8;
|
|
use serde::Deserialize;
|
|
use std::collections::HashMap;
|
|
use std::mem::MaybeUninit;
|
|
use std::ops::Deref;
|
|
use std::ops::DerefMut;
|
|
use std::ptr;
|
|
|
|
pub struct CoverageCollector {
|
|
v8_channel: v8::inspector::ChannelBase,
|
|
v8_session: v8::UniqueRef<v8::inspector::V8InspectorSession>,
|
|
response_map: HashMap<i32, oneshot::Sender<serde_json::Value>>,
|
|
next_message_id: i32,
|
|
}
|
|
|
|
impl Deref for CoverageCollector {
|
|
type Target = v8::inspector::V8InspectorSession;
|
|
fn deref(&self) -> &Self::Target {
|
|
&self.v8_session
|
|
}
|
|
}
|
|
|
|
impl DerefMut for CoverageCollector {
|
|
fn deref_mut(&mut self) -> &mut Self::Target {
|
|
&mut self.v8_session
|
|
}
|
|
}
|
|
|
|
impl v8::inspector::ChannelImpl for CoverageCollector {
|
|
fn base(&self) -> &v8::inspector::ChannelBase {
|
|
&self.v8_channel
|
|
}
|
|
|
|
fn base_mut(&mut self) -> &mut v8::inspector::ChannelBase {
|
|
&mut self.v8_channel
|
|
}
|
|
|
|
fn send_response(
|
|
&mut self,
|
|
call_id: i32,
|
|
message: v8::UniquePtr<v8::inspector::StringBuffer>,
|
|
) {
|
|
let raw_message = message.unwrap().string().to_string();
|
|
let message = serde_json::from_str(&raw_message).unwrap();
|
|
self
|
|
.response_map
|
|
.remove(&call_id)
|
|
.unwrap()
|
|
.send(message)
|
|
.unwrap();
|
|
}
|
|
|
|
fn send_notification(
|
|
&mut self,
|
|
_message: v8::UniquePtr<v8::inspector::StringBuffer>,
|
|
) {
|
|
}
|
|
|
|
fn flush_protocol_notifications(&mut self) {}
|
|
}
|
|
|
|
impl CoverageCollector {
|
|
const CONTEXT_GROUP_ID: i32 = 1;
|
|
|
|
pub fn new(inspector_ptr: *mut DenoInspector) -> Box<Self> {
|
|
new_box_with(move |self_ptr| {
|
|
let v8_channel = v8::inspector::ChannelBase::new::<Self>();
|
|
let v8_session = unsafe { &mut *inspector_ptr }.connect(
|
|
Self::CONTEXT_GROUP_ID,
|
|
unsafe { &mut *self_ptr },
|
|
v8::inspector::StringView::empty(),
|
|
);
|
|
|
|
let response_map = HashMap::new();
|
|
let next_message_id = 0;
|
|
|
|
Self {
|
|
v8_channel,
|
|
v8_session,
|
|
response_map,
|
|
next_message_id,
|
|
}
|
|
})
|
|
}
|
|
|
|
async fn post_message(
|
|
&mut self,
|
|
method: String,
|
|
params: Option<serde_json::Value>,
|
|
) -> Result<serde_json::Value, AnyError> {
|
|
let id = self.next_message_id;
|
|
self.next_message_id += 1;
|
|
|
|
let (sender, receiver) = oneshot::channel::<serde_json::Value>();
|
|
self.response_map.insert(id, sender);
|
|
|
|
let message = json!({
|
|
"id": id,
|
|
"method": method,
|
|
"params": params,
|
|
});
|
|
|
|
let raw_message = serde_json::to_string(&message).unwrap();
|
|
let raw_message = v8::inspector::StringView::from(raw_message.as_bytes());
|
|
self.v8_session.dispatch_protocol_message(raw_message);
|
|
|
|
let response = receiver.await.unwrap();
|
|
if let Some(error) = response.get("error") {
|
|
return Err(generic_error(format!("{}", error)));
|
|
}
|
|
|
|
let result = response.get("result").unwrap().clone();
|
|
Ok(result)
|
|
}
|
|
|
|
pub async fn start_collecting(&mut self) -> Result<(), AnyError> {
|
|
self
|
|
.post_message("Debugger.enable".to_string(), None)
|
|
.await?;
|
|
|
|
self
|
|
.post_message("Runtime.enable".to_string(), None)
|
|
.await?;
|
|
|
|
self
|
|
.post_message("Profiler.enable".to_string(), None)
|
|
.await?;
|
|
|
|
self
|
|
.post_message(
|
|
"Profiler.startPreciseCoverage".to_string(),
|
|
Some(json!({"callCount": true, "detailed": true})),
|
|
)
|
|
.await?;
|
|
|
|
Ok(())
|
|
}
|
|
|
|
pub async fn collect(&mut self) -> Result<Vec<Coverage>, AnyError> {
|
|
let result = self
|
|
.post_message("Profiler.takePreciseCoverage".to_string(), None)
|
|
.await?;
|
|
|
|
let take_coverage_result: TakePreciseCoverageResult =
|
|
serde_json::from_value(result)?;
|
|
|
|
let mut coverages: Vec<Coverage> = Vec::new();
|
|
for script_coverage in take_coverage_result.result {
|
|
let result = self
|
|
.post_message(
|
|
"Debugger.getScriptSource".to_string(),
|
|
Some(json!({
|
|
"scriptId": script_coverage.script_id,
|
|
})),
|
|
)
|
|
.await?;
|
|
|
|
let get_script_source_result: GetScriptSourceResult =
|
|
serde_json::from_value(result)?;
|
|
|
|
coverages.push(Coverage {
|
|
script_coverage,
|
|
script_source: get_script_source_result.script_source,
|
|
})
|
|
}
|
|
|
|
Ok(coverages)
|
|
}
|
|
|
|
pub async fn stop_collecting(&mut self) -> Result<(), AnyError> {
|
|
self
|
|
.post_message("Profiler.stopPreciseCoverage".to_string(), None)
|
|
.await?;
|
|
self
|
|
.post_message("Profiler.disable".to_string(), None)
|
|
.await?;
|
|
self
|
|
.post_message("Runtime.disable".to_string(), None)
|
|
.await?;
|
|
self
|
|
.post_message("Debugger.disable".to_string(), None)
|
|
.await?;
|
|
|
|
Ok(())
|
|
}
|
|
}
|
|
|
|
#[derive(Debug, Deserialize)]
|
|
#[serde(rename_all = "camelCase")]
|
|
pub struct CoverageRange {
|
|
pub start_offset: usize,
|
|
pub end_offset: usize,
|
|
pub count: usize,
|
|
}
|
|
|
|
#[derive(Debug, Deserialize)]
|
|
#[serde(rename_all = "camelCase")]
|
|
pub struct FunctionCoverage {
|
|
pub function_name: String,
|
|
pub ranges: Vec<CoverageRange>,
|
|
pub is_block_coverage: bool,
|
|
}
|
|
|
|
#[derive(Debug, Deserialize)]
|
|
#[serde(rename_all = "camelCase")]
|
|
pub struct ScriptCoverage {
|
|
pub script_id: String,
|
|
pub url: String,
|
|
pub functions: Vec<FunctionCoverage>,
|
|
}
|
|
|
|
#[derive(Debug, Deserialize)]
|
|
#[serde(rename_all = "camelCase")]
|
|
pub struct Coverage {
|
|
pub script_coverage: ScriptCoverage,
|
|
pub script_source: String,
|
|
}
|
|
|
|
#[derive(Debug, Deserialize)]
|
|
#[serde(rename_all = "camelCase")]
|
|
struct TakePreciseCoverageResult {
|
|
result: Vec<ScriptCoverage>,
|
|
}
|
|
|
|
#[derive(Debug, Deserialize)]
|
|
#[serde(rename_all = "camelCase")]
|
|
pub struct GetScriptSourceResult {
|
|
pub script_source: String,
|
|
pub bytecode: Option<String>,
|
|
}
|
|
|
|
pub struct PrettyCoverageReporter {
|
|
coverages: Vec<Coverage>,
|
|
}
|
|
|
|
// TODO(caspervonb) add support for lcov output (see geninfo(1) for format spec).
|
|
impl PrettyCoverageReporter {
|
|
pub fn new(coverages: Vec<Coverage>) -> PrettyCoverageReporter {
|
|
PrettyCoverageReporter { coverages }
|
|
}
|
|
|
|
pub fn get_report(&self) -> String {
|
|
let mut report = String::from("test coverage:\n");
|
|
|
|
for coverage in &self.coverages {
|
|
if let Some(coverage_report) = Self::get_coverage_report(coverage) {
|
|
report.push_str(&format!("{}\n", coverage_report))
|
|
}
|
|
}
|
|
|
|
report
|
|
}
|
|
|
|
fn get_coverage_report(coverage: &Coverage) -> Option<String> {
|
|
let mut total_lines = 0;
|
|
let mut covered_lines = 0;
|
|
|
|
let mut line_offset = 0;
|
|
|
|
for line in coverage.script_source.lines() {
|
|
let line_start_offset = line_offset;
|
|
let line_end_offset = line_start_offset + line.len();
|
|
|
|
let mut count = 0;
|
|
for function in &coverage.script_coverage.functions {
|
|
for range in &function.ranges {
|
|
if range.start_offset <= line_start_offset
|
|
&& range.end_offset >= line_end_offset
|
|
{
|
|
count += range.count;
|
|
if range.count == 0 {
|
|
count = 0;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if count > 0 {
|
|
covered_lines += 1;
|
|
}
|
|
|
|
total_lines += 1;
|
|
line_offset += line.len();
|
|
}
|
|
|
|
let line_ratio = covered_lines as f32 / total_lines as f32;
|
|
let line_coverage = format!("{:.3}%", line_ratio * 100.0);
|
|
|
|
let line = if line_ratio >= 0.9 {
|
|
format!(
|
|
"{} {}",
|
|
coverage.script_coverage.url,
|
|
colors::green(&line_coverage)
|
|
)
|
|
} else if line_ratio >= 0.75 {
|
|
format!(
|
|
"{} {}",
|
|
coverage.script_coverage.url,
|
|
colors::yellow(&line_coverage)
|
|
)
|
|
} else {
|
|
format!(
|
|
"{} {}",
|
|
coverage.script_coverage.url,
|
|
colors::red(&line_coverage)
|
|
)
|
|
};
|
|
|
|
Some(line)
|
|
}
|
|
}
|
|
|
|
fn new_box_with<T>(new_fn: impl FnOnce(*mut T) -> T) -> Box<T> {
|
|
let b = Box::new(MaybeUninit::<T>::uninit());
|
|
let p = Box::into_raw(b) as *mut T;
|
|
unsafe { ptr::write(p, new_fn(p)) };
|
|
unsafe { Box::from_raw(p) }
|
|
}
|
|
|
|
pub fn filter_script_coverages(
|
|
coverages: Vec<Coverage>,
|
|
test_file_url: Url,
|
|
test_modules: Vec<Url>,
|
|
) -> Vec<Coverage> {
|
|
coverages
|
|
.into_iter()
|
|
.filter(|e| {
|
|
if let Ok(url) = Url::parse(&e.script_coverage.url) {
|
|
if url.path().ends_with("__anonymous__") {
|
|
return false;
|
|
}
|
|
|
|
if url == test_file_url {
|
|
return false;
|
|
}
|
|
|
|
for test_module_url in &test_modules {
|
|
if &url == test_module_url {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
if let Ok(path) = url.to_file_path() {
|
|
for test_module_url in &test_modules {
|
|
if let Ok(test_module_path) = test_module_url.to_file_path() {
|
|
if path.starts_with(test_module_path.parent().unwrap()) {
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
false
|
|
})
|
|
.collect::<Vec<Coverage>>()
|
|
}
|