1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
//! Handling notifications and request received from neovim
//!
//! The core of a plugin is defining and implementing the
//! [`handler`](crate::rpc::handler::Handler).
use std::{marker::PhantomData, sync::Arc};

use async_trait::async_trait;
use futures::io::AsyncWrite;
use rmpv::Value;

use crate::Neovim;

/// The central functionality of a plugin. The trait bounds asure that each
/// asynchronous task can receive a copy of the handler, so some state can be
/// shared.
#[async_trait]
pub trait Handler: Send + Sync + Clone + 'static {
  /// The type where we write our responses to requests. Handling of incoming
  /// requests/notifications is done on the io loop, which passes the parsed
  /// messages to the handler.
  type Writer: AsyncWrite + Send + Unpin + 'static;

  /// Handling an rpc request. The ID's of requests are handled by the
  /// [`neovim`](crate::neovim::Neovim) instance.
  async fn handle_request(
    &self,
    _name: String,
    _args: Vec<Value>,
    _neovim: Neovim<Self::Writer>,
  ) -> Result<Value, Value> {
    Err(Value::from("Not implemented"))
  }

  /// Handling an rpc notification. Notifications are handled one at a time in
  /// the order in which they were received, and will block new requests from
  /// being received until handle_notify returns.
  async fn handle_notify(
    &self,
    _name: String,
    _args: Vec<Value>,
    _neovim: Neovim<<Self as Handler>::Writer>,
  ) {
  }
}

/// The dummy handler defaults to doing nothing with a notification, and
/// returning a generic error for a request. It can be used if a plugin only
/// wants to send requests to neovim and get responses, but not handle any
/// notifications or requests.
#[derive(Default)]
pub struct Dummy<Q>
where
  Q: AsyncWrite + Send + Sync + Unpin + 'static,
{
  q: Arc<PhantomData<Q>>,
}

impl<Q> Clone for Dummy<Q>
where
  Q: AsyncWrite + Send + Sync + Unpin + 'static,
{
  fn clone(&self) -> Self {
    Dummy { q: self.q.clone() }
  }
}

impl<Q> Handler for Dummy<Q>
where
  Q: AsyncWrite + Send + Sync + Unpin + 'static,
{
  type Writer = Q;
}

impl<Q> Dummy<Q>
where
  Q: AsyncWrite + Send + Sync + Unpin + 'static,
{
  #[must_use]
  pub fn new() -> Dummy<Q> {
    Dummy {
      q: Arc::new(PhantomData),
    }
  }
}