mirror of
https://github.com/zed-industries/zed.git
synced 2025-02-03 08:54:04 +00:00
748598e419
Co-Authored-By: Nathan Sobo <nathan@zed.dev>
169 lines
4.3 KiB
Rust
169 lines
4.3 KiB
Rust
#[cfg(feature = "test-support")]
|
|
pub mod test;
|
|
|
|
use futures::Future;
|
|
use std::{
|
|
cmp::Ordering,
|
|
pin::Pin,
|
|
task::{Context, Poll},
|
|
};
|
|
|
|
pub fn post_inc(value: &mut usize) -> usize {
|
|
let prev = *value;
|
|
*value += 1;
|
|
prev
|
|
}
|
|
|
|
/// Extend a sorted vector with a sorted sequence of items, maintaining the vector's sort order and
|
|
/// enforcing a maximum length. Sort the items according to the given callback. Before calling this,
|
|
/// both `vec` and `new_items` should already be sorted according to the `cmp` comparator.
|
|
pub fn extend_sorted<T, I, F>(vec: &mut Vec<T>, new_items: I, limit: usize, mut cmp: F)
|
|
where
|
|
I: IntoIterator<Item = T>,
|
|
F: FnMut(&T, &T) -> Ordering,
|
|
{
|
|
let mut start_index = 0;
|
|
for new_item in new_items {
|
|
if let Err(i) = vec[start_index..].binary_search_by(|m| cmp(m, &new_item)) {
|
|
let index = start_index + i;
|
|
if vec.len() < limit {
|
|
vec.insert(index, new_item);
|
|
} else if index < vec.len() {
|
|
vec.pop();
|
|
vec.insert(index, new_item);
|
|
}
|
|
start_index = index;
|
|
}
|
|
}
|
|
}
|
|
pub trait ResultExt {
|
|
type Ok;
|
|
|
|
fn log_err(self) -> Option<Self::Ok>;
|
|
fn warn_on_err(self) -> Option<Self::Ok>;
|
|
}
|
|
|
|
impl<T> ResultExt for anyhow::Result<T> {
|
|
type Ok = T;
|
|
|
|
fn log_err(self) -> Option<T> {
|
|
match self {
|
|
Ok(value) => Some(value),
|
|
Err(error) => {
|
|
log::error!("{:?}", error);
|
|
None
|
|
}
|
|
}
|
|
}
|
|
|
|
fn warn_on_err(self) -> Option<T> {
|
|
match self {
|
|
Ok(value) => Some(value),
|
|
Err(error) => {
|
|
log::warn!("{:?}", error);
|
|
None
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
pub trait TryFutureExt {
|
|
fn log_err(self) -> LogErrorFuture<Self>
|
|
where
|
|
Self: Sized;
|
|
fn warn_on_err(self) -> LogErrorFuture<Self>
|
|
where
|
|
Self: Sized;
|
|
}
|
|
|
|
impl<F, T> TryFutureExt for F
|
|
where
|
|
F: Future<Output = anyhow::Result<T>>,
|
|
{
|
|
fn log_err(self) -> LogErrorFuture<Self>
|
|
where
|
|
Self: Sized,
|
|
{
|
|
LogErrorFuture(self, log::Level::Error)
|
|
}
|
|
|
|
fn warn_on_err(self) -> LogErrorFuture<Self>
|
|
where
|
|
Self: Sized,
|
|
{
|
|
LogErrorFuture(self, log::Level::Warn)
|
|
}
|
|
}
|
|
|
|
pub struct LogErrorFuture<F>(F, log::Level);
|
|
|
|
impl<F, T> Future for LogErrorFuture<F>
|
|
where
|
|
F: Future<Output = anyhow::Result<T>>,
|
|
{
|
|
type Output = Option<T>;
|
|
|
|
fn poll(self: std::pin::Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Self::Output> {
|
|
let level = self.1;
|
|
let inner = unsafe { Pin::new_unchecked(&mut self.get_unchecked_mut().0) };
|
|
match inner.poll(cx) {
|
|
Poll::Ready(output) => Poll::Ready(match output {
|
|
Ok(output) => Some(output),
|
|
Err(error) => {
|
|
log::log!(level, "{:?}", error);
|
|
None
|
|
}
|
|
}),
|
|
Poll::Pending => Poll::Pending,
|
|
}
|
|
}
|
|
}
|
|
|
|
#[cfg(test)]
|
|
mod tests {
|
|
use super::*;
|
|
|
|
#[test]
|
|
fn test_extend_sorted() {
|
|
let mut vec = vec![];
|
|
|
|
extend_sorted(&mut vec, vec![21, 17, 13, 8, 1, 0], 5, |a, b| b.cmp(a));
|
|
assert_eq!(vec, &[21, 17, 13, 8, 1]);
|
|
|
|
extend_sorted(&mut vec, vec![101, 19, 17, 8, 2], 8, |a, b| b.cmp(a));
|
|
assert_eq!(vec, &[101, 21, 19, 17, 13, 8, 2, 1]);
|
|
|
|
extend_sorted(&mut vec, vec![1000, 19, 17, 9, 5], 8, |a, b| b.cmp(a));
|
|
assert_eq!(vec, &[1000, 101, 21, 19, 17, 13, 9, 8]);
|
|
}
|
|
}
|
|
|
|
// Allow surf Results to accept context like other Results do when
|
|
// using anyhow.
|
|
pub trait SurfResultExt {
|
|
fn context<C>(self, cx: C) -> Self
|
|
where
|
|
C: std::fmt::Display + Send + Sync + 'static;
|
|
|
|
fn with_context<C, F>(self, f: F) -> Self
|
|
where
|
|
C: std::fmt::Display + Send + Sync + 'static,
|
|
F: FnOnce() -> C;
|
|
}
|
|
|
|
impl<T> SurfResultExt for surf::Result<T> {
|
|
fn context<C>(self, cx: C) -> Self
|
|
where
|
|
C: std::fmt::Display + Send + Sync + 'static,
|
|
{
|
|
self.map_err(|e| surf::Error::new(e.status(), e.into_inner().context(cx)))
|
|
}
|
|
|
|
fn with_context<C, F>(self, f: F) -> Self
|
|
where
|
|
C: std::fmt::Display + Send + Sync + 'static,
|
|
F: FnOnce() -> C,
|
|
{
|
|
self.map_err(|e| surf::Error::new(e.status(), e.into_inner().context(f())))
|
|
}
|
|
}
|