Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Update test callable discovery to work within the language service #2095

Open
wants to merge 62 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
62 commits
Select commit Hold shift + click to select a range
b79c31e
initial work on testing
sezna Dec 10, 2024
1c09783
Merge branch 'main' of github.com:microsoft/qsharp into alex/testHarness
sezna Dec 10, 2024
54eb209
progress on test explorer
sezna Dec 10, 2024
e6a0941
test collection works
sezna Dec 10, 2024
0495a6d
tests run
sezna Dec 10, 2024
eae1b4d
add todos
sezna Dec 10, 2024
759036f
switching to namespaces included with callable names
sezna Dec 10, 2024
cec4bf9
scoped test names
sezna Dec 10, 2024
ad7d3e2
deduplicate parent items
sezna Dec 12, 2024
d9f288d
make running child items work
sezna Dec 12, 2024
9ec9f41
auto-refresh test cases
sezna Dec 12, 2024
a03ed14
wip -- checkpoint
sezna Dec 12, 2024
c11476a
wip
sezna Dec 12, 2024
b247c35
Remove codelens stuff
sezna Dec 12, 2024
e75f65b
update libraries to use new testing harness
sezna Dec 12, 2024
131a341
remove bad imports
sezna Dec 12, 2024
b52ad25
document some functions
sezna Dec 12, 2024
180368e
Add comment
sezna Dec 12, 2024
e960916
Remove todos
sezna Dec 12, 2024
7d2aadc
Fix nested tests
sezna Dec 12, 2024
ac8bfd7
initial round of PR feedback
sezna Dec 13, 2024
c22d14d
move discovery of test items into compiler layer
sezna Dec 13, 2024
5a28ef1
Use a pass to detect test attribute errors and report them nicely
sezna Dec 16, 2024
16a2aed
use getActiveProgram
sezna Dec 16, 2024
214097c
remove unnecessary api in main.ts
sezna Dec 16, 2024
adbc4b2
Fmt
sezna Dec 16, 2024
5b6a1f8
fix lints
sezna Dec 16, 2024
7cee532
update tests
sezna Dec 16, 2024
be64105
filter out invalid test items
sezna Dec 16, 2024
e34b7d2
Merge branch 'main' of github.com:microsoft/qsharp into alex/testHarness
sezna Dec 18, 2024
fa1ca42
wip: start to add locations to the return type for test callables
sezna Dec 18, 2024
790c29b
get spans/ranges hooked up
sezna Dec 18, 2024
9d0190c
abstract compiler worker generation into a common singleton worker
sezna Dec 20, 2024
38e0f4b
wipz
sezna Dec 20, 2024
43f4e17
use updateDocument events for test discovery
sezna Dec 20, 2024
bcd6d34
it works, but i'd rather not have the tests collapse on auto refresh
sezna Dec 23, 2024
5421cb2
Fmt
sezna Dec 23, 2024
804fb0b
rename Vscode to VsCode
sezna Dec 23, 2024
ffab724
rename collectTestCallables to getTestCallables
sezna Dec 23, 2024
cca48b5
switch to debug event target; remove unnecessary result
sezna Dec 23, 2024
eea9581
rename test explorer to test discovery
sezna Dec 23, 2024
540de0b
fmt
sezna Dec 23, 2024
f171141
update tests for test attribute
sezna Dec 31, 2024
8e3c9a4
wip
sezna Jan 2, 2025
f18dc39
wip: refactor: it works
sezna Jan 2, 2025
cf35b19
Remove comment
sezna Jan 2, 2025
36abbd1
wip
sezna Jan 2, 2025
bf8b277
wip
sezna Jan 2, 2025
ff31000
individual test parity tracking works
sezna Jan 7, 2025
b72ca75
Rename from parity to version
sezna Jan 7, 2025
f52330d
pull test explorer out into its own file
sezna Jan 7, 2025
865cce1
lint fixes
sezna Jan 7, 2025
1f88c97
rust lint updates
sezna Jan 7, 2025
e0347c6
update expect test
sezna Jan 7, 2025
468345b
update test case for new test callables
sezna Jan 7, 2025
00d5f4f
basics.js test
sezna Jan 7, 2025
57bc6c1
add basics.js tests
sezna Jan 7, 2025
d1267c4
undo common compiler worker
sezna Jan 7, 2025
8255955
Merge branch 'main' of github.com:microsoft/qsharp into alex/testHarness
sezna Jan 7, 2025
007a76d
Merge branch 'alex/testHarness' of github.com:microsoft/qsharp into a…
sezna Jan 7, 2025
1d5ef8e
Fix output
sezna Jan 9, 2025
6ca81d0
mid PR review feedback
sezna Jan 9, 2025
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
12 changes: 12 additions & 0 deletions compiler/qsc_frontend/src/lower.rs
Original file line number Diff line number Diff line change
Expand Up @@ -443,6 +443,18 @@ impl With<'_> {
None
}
},
Ok(hir::Attr::Test) => {
// verify that no args are passed to the attribute
match &*attr.arg.kind {
ast::ExprKind::Tuple(args) if args.is_empty() => {}
_ => {
self.lowerer
.errors
.push(Error::InvalidAttrArgs("()".to_string(), attr.arg.span));
}
}
Some(hir::Attr::Test)
}
Err(()) => {
self.lowerer.errors.push(Error::UnknownAttr(
attr.name.name.to_string(),
Expand Down
55 changes: 55 additions & 0 deletions compiler/qsc_hir/src/hir.rs
Original file line number Diff line number Diff line change
Expand Up @@ -279,6 +279,57 @@ impl Display for Package {
}
}

/// The name of a test callable, including its parent namespace.
pub type TestCallableName = String;

impl Package {
/// Returns a collection of the fully qualified names of any callables annotated with `@Test()`
pub fn get_test_callables(&self) -> Vec<(TestCallableName, Span)> {
let items_with_test_attribute = self
.items
.iter()
.filter(|(_, item)| item.attrs.iter().any(|attr| *attr == Attr::Test));

let callables = items_with_test_attribute
.filter(|(_, item)| matches!(item.kind, ItemKind::Callable(_)));

let callable_names = callables
.filter_map(|(_, item)| -> Option<_> {
if let ItemKind::Callable(callable) = &item.kind {
if !callable.generics.is_empty()
|| callable.input.kind != PatKind::Tuple(vec![])
{
return None;
}
// this is indeed a test callable, so let's grab its parent name
let name = match item.parent {
None => Default::default(),
Some(parent_id) => {
let parent_item = self
.items
.get(parent_id)
.expect("Parent item did not exist in package");
if let ItemKind::Namespace(ns, _) = &parent_item.kind {
format!("{}.{}", ns.name(), callable.name.name)
} else {
callable.name.name.to_string()
}
}
};

let span = item.span;

Some((name, span))
} else {
None
}
})
.collect::<Vec<_>>();

callable_names
}
}

/// An item.
#[derive(Clone, Debug, PartialEq)]
pub struct Item {
Expand Down Expand Up @@ -1359,6 +1410,8 @@ pub enum Attr {
/// Indicates that an intrinsic callable is a reset. This means that the operation will be marked as
/// "irreversible" in the generated QIR.
Reset,
/// Indicates that a callable is a test case.
Test,
}

impl Attr {
Expand All @@ -1376,6 +1429,7 @@ The `not` operator is also supported to negate the attribute, e.g. `not Adaptive
Attr::SimulatableIntrinsic => "Indicates that an item should be treated as an intrinsic callable for QIR code generation and any implementation should only be used during simulation.",
Attr::Measurement => "Indicates that an intrinsic callable is a measurement. This means that the operation will be marked as \"irreversible\" in the generated QIR, and output Result types will be moved to the arguments.",
Attr::Reset => "Indicates that an intrinsic callable is a reset. This means that the operation will be marked as \"irreversible\" in the generated QIR.",
Attr::Test => "Indicates that a callable is a test case.",
}
}
}
Expand All @@ -1391,6 +1445,7 @@ impl FromStr for Attr {
"SimulatableIntrinsic" => Ok(Self::SimulatableIntrinsic),
"Measurement" => Ok(Self::Measurement),
"Reset" => Ok(Self::Reset),
"Test" => Ok(Self::Test),
_ => Err(()),
}
}
Expand Down
5 changes: 4 additions & 1 deletion compiler/qsc_lowerer/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -943,7 +943,10 @@ fn lower_attrs(attrs: &[hir::Attr]) -> Vec<fir::Attr> {
hir::Attr::EntryPoint => Some(fir::Attr::EntryPoint),
hir::Attr::Measurement => Some(fir::Attr::Measurement),
hir::Attr::Reset => Some(fir::Attr::Reset),
hir::Attr::SimulatableIntrinsic | hir::Attr::Unimplemented | hir::Attr::Config => None,
hir::Attr::SimulatableIntrinsic
| hir::Attr::Unimplemented
| hir::Attr::Config
| hir::Attr::Test => None,
})
.collect()
}
Expand Down
17 changes: 17 additions & 0 deletions compiler/qsc_parse/src/item/tests.rs
Original file line number Diff line number Diff line change
Expand Up @@ -2396,3 +2396,20 @@ fn top_level_nodes_error_recovery() {
]"#]],
);
}

#[test]
fn test_attribute() {
check(
parse,
"@Test() function Foo() : Unit {}",
&expect![[r#"
Item _id_ [0-32]:
Attr _id_ [0-7] (Ident _id_ [1-5] "Test"):
Expr _id_ [5-7]: Unit
Callable _id_ [8-32] (Function):
name: Ident _id_ [17-20] "Foo"
input: Pat _id_ [20-22]: Unit
output: Type _id_ [25-29]: Path: Path _id_ [25-29] (Ident _id_ [25-29] "Unit")
body: Block: Block _id_ [30-32]: <empty>"#]],
);
}
6 changes: 6 additions & 0 deletions compiler/qsc_passes/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,7 @@ mod measurement;
mod replace_qubit_allocation;
mod reset;
mod spec_gen;
mod test_attribute;

use callable_limits::CallableLimits;
use capabilitiesck::{check_supported_capabilities, lower_store, run_rca_pass};
Expand Down Expand Up @@ -52,6 +53,7 @@ pub enum Error {
Measurement(measurement::Error),
Reset(reset::Error),
SpecGen(spec_gen::Error),
TestAttribute(test_attribute::TestAttributeError),
}

#[derive(Clone, Copy, Debug, PartialEq)]
Expand Down Expand Up @@ -121,6 +123,9 @@ impl PassContext {
ReplaceQubitAllocation::new(core, assigner).visit_package(package);
Validator::default().visit_package(package);

let test_attribute_errors = test_attribute::validate_test_attributes(package);
Validator::default().visit_package(package);

callable_errors
.into_iter()
.map(Error::CallableLimits)
Expand All @@ -130,6 +135,7 @@ impl PassContext {
.chain(entry_point_errors)
.chain(measurement_decl_errors.into_iter().map(Error::Measurement))
.chain(reset_decl_errors.into_iter().map(Error::Reset))
.chain(test_attribute_errors.into_iter().map(Error::TestAttribute))
.collect()
}

Expand Down
47 changes: 47 additions & 0 deletions compiler/qsc_passes/src/test_attribute.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,47 @@
// Copyright (c) Microsoft Corporation.
// Licensed under the MIT License.

use miette::Diagnostic;
use qsc_data_structures::span::Span;
use qsc_hir::{hir::Attr, visit::Visitor};
use thiserror::Error;

#[cfg(test)]
mod tests;

#[derive(Clone, Debug, Diagnostic, Error)]
pub enum TestAttributeError {
#[error("test callables cannot take arguments")]
CallableHasParameters(#[label] Span),
#[error("test callables cannot have type parameters")]
CallableHasTypeParameters(#[label] Span),
}

pub(crate) fn validate_test_attributes(
package: &mut qsc_hir::hir::Package,
) -> Vec<TestAttributeError> {
let mut validator = TestAttributeValidator { errors: Vec::new() };
validator.visit_package(package);
validator.errors
}

struct TestAttributeValidator {
errors: Vec<TestAttributeError>,
}

impl<'a> Visitor<'a> for TestAttributeValidator {
fn visit_callable_decl(&mut self, decl: &'a qsc_hir::hir::CallableDecl) {
if decl.attrs.iter().any(|attr| matches!(attr, Attr::Test)) {
if !decl.generics.is_empty() {
self.errors
.push(TestAttributeError::CallableHasTypeParameters(
decl.name.span,
));
}
if decl.input.ty != qsc_hir::ty::Ty::UNIT {
self.errors
.push(TestAttributeError::CallableHasParameters(decl.name.span));
Comment on lines +38 to +43
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Instead of just the name, could you use the whole signature's span, so that the squiggles appear below the problematic code and not the name?

And now that I wrote that... I realize there may not be an easy way to get that span from the AST. I think I hit this issue before. Hmm...

}
}
}
}
110 changes: 110 additions & 0 deletions compiler/qsc_passes/src/test_attribute/tests.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,110 @@
// Copyright (c) Microsoft Corporation.
// Licensed under the MIT License.

use expect_test::{expect, Expect};
use indoc::indoc;
use qsc_data_structures::{language_features::LanguageFeatures, target::TargetCapabilityFlags};
use qsc_frontend::compile::{self, compile, PackageStore, SourceMap};
use qsc_hir::{validate::Validator, visit::Visitor};

use crate::test_attribute::validate_test_attributes;

fn check(file: &str, expect: &Expect) {
let store = PackageStore::new(compile::core());
let sources = SourceMap::new([("test".into(), file.into())], None);
let mut unit = compile(
&store,
&[],
sources,
TargetCapabilityFlags::all(),
LanguageFeatures::default(),
);
assert!(unit.errors.is_empty(), "{:?}", unit.errors);

let errors = validate_test_attributes(&mut unit.package);
Validator::default().visit_package(&unit.package);
if errors.is_empty() {
expect.assert_eq(&unit.package.to_string());
} else {
expect.assert_debug_eq(&errors);
}
}

#[test]
fn callable_cant_have_params() {
check(
indoc! {"
namespace test {
@Test()
operation A(q : Qubit) : Unit {

}
}
"},
&expect![[r#"
[
CallableHasParameters(
Span {
lo: 43,
hi: 44,
},
),
]
"#]],
);
}

#[test]
fn callable_cant_have_type_params() {
check(
indoc! {"
namespace test {
@Test()
operation A<'T>() : Unit {

}
}
"},
&expect![[r#"
[
CallableHasTypeParameters(
Span {
lo: 43,
hi: 44,
},
),
]
"#]],
);
}

#[test]
fn callable_is_valid_test_callable() {
check(
indoc! {"
namespace test {
@Test()
operation A() : Unit {

}
}
"},
&expect![[r#"
Package:
Item 0 [0-64] (Public):
Namespace (Ident 5 [10-14] "test"): Item 1
Item 1 [21-62] (Internal):
Parent: 0
Test
Callable 0 [33-62] (operation):
name: Ident 1 [43-44] "A"
input: Pat 2 [44-46] [Type Unit]: Unit
output: Unit
functors: empty set
body: SpecDecl 3 [33-62]: Impl:
Block 4 [54-62]: <empty>
adj: <none>
ctl: <none>
ctl-adj: <none>"#]],
);
}
7 changes: 7 additions & 0 deletions language_service/src/compilation.rs
Original file line number Diff line number Diff line change
Expand Up @@ -37,6 +37,7 @@ pub(crate) struct Compilation {
pub compile_errors: Vec<compile::Error>,
pub kind: CompilationKind,
pub dependencies: FxHashMap<PackageId, Option<PackageAlias>>,
pub test_cases: Vec<(String, Span)>,
}

#[derive(Debug)]
Expand Down Expand Up @@ -102,6 +103,8 @@ impl Compilation {

run_linter_passes(&mut compile_errors, &package_store, unit, lints_config);

let test_cases = unit.package.get_test_callables();

Self {
package_store,
user_package_id: package_id,
Expand All @@ -111,6 +114,7 @@ impl Compilation {
compile_errors,
project_errors,
dependencies: user_code_dependencies.into_iter().collect(),
test_cases,
}
}

Expand Down Expand Up @@ -218,12 +222,15 @@ impl Compilation {
.chain(once((source_package_id, None)))
.collect();

let test_cases = unit.package.get_test_callables();

Self {
package_store,
user_package_id: package_id,
compile_errors: errors,
project_errors: project.as_ref().map_or_else(Vec::new, |p| p.errors.clone()),
kind: CompilationKind::Notebook { project },
test_cases,
dependencies,
}
}
Expand Down
1 change: 1 addition & 0 deletions language_service/src/completion.rs
Original file line number Diff line number Diff line change
Expand Up @@ -165,6 +165,7 @@ fn collect_hardcoded_words(expected: WordKinds) -> Vec<Completion> {
),
Completion::new("Measurement".to_string(), CompletionItemKind::Interface),
Completion::new("Reset".to_string(), CompletionItemKind::Interface),
Completion::new("Test".to_string(), CompletionItemKind::Interface),
]);
}
HardcodedIdentKind::Size => {
Expand Down
Loading
Loading