Struct TryElection

Source
pub(crate) struct TryElection<T: Config> {
    pub by: T::Public,
    pub at: BlockNumberFor<T>,
}
Expand description

Operational context for election transaction execution.

This type represents the contextual information required to attempt authors election using an affidavit key during the routine lifecycle.

Election attempts are optimistic, sequential routines that may execute repeatedly as part of a looped execution model (e.g. via offchain workers). Although election typically follows the affidavit declaration phase, explicit looping of sequential routines allows this phase to run earlier. The routine remains safe, as successful election is gated by retrievable storage state, election window constraints, and eligibility checks.

§Notes

  • This is not a transaction payload.
  • This type is never submitted on-chain.
  • It is used internally during:
    • fetching the active-tagged affidavit key (storage-reference + keystore-pair)
    • election payload signing using the affidavit key pair
    • submission of the elect extrinsic
    • retry-driven OCW election execution

§Fork Awareness

  • The at field captures the block number at which the election attempt begins.
  • When executed via offchain workers, this context must tolerate forks, re-orgs, and speculative execution.
  • Implementors are responsible for ensuring idempotency and re-entrancy safety.

§Dependency: InitAffidavitKey

This routine depends on the affidavit key initialization phase. It assumes that a active-tagged (offchain storage referenced) affidavit key and its pair in crypto-store is available. Whenever this invariant is violated (e.g. missing key, failed signing, or inconsistent state), control is redirected to InitAffidavitKey to repair and re-validate the key before retrying the election attempt.

§Election Flow

The routine fetches the active-tagged affidavit key-pair (referenced via offchain storage and retrieved through local keystore), then attempts author election. It short-circuits when outside the election window, when ineligible, or when already elected (to avoid redundancy). Failures or inconsistencies re-enter the affidavit initialization phase for re-validation and repair.

loop {
    // Ensure active-tagged affidavit key is available
    InitAffidavitKey::ensure_active_affidavit_key();

    // Fetch affidavit key pair (offchain storage reference + keystore-pair)
    let key_pair = fetch_affidavit_key_pair();

    if !within_election_window()
        || !eligible_to_elect()
        || already_elected()
    {
        break; // proceed to declaration phase (no election required)
    }

    let payload = Default::default();

    match sign_payload_with(key_pair)
        .and_then(|payload| submit_elect_authors_extrinsic(payload))
    {
        Ok(_) => break,     // elect-phase completed successfully
        Err(_) => continue, // retry via initialization phase again
    }
}

§Guarantees

This routine proceeds to the affidavit declaration phase under exactly two conditions:

  1. Initialization-only path: a active-tagged affidavit key-pair exists, but election constraints are not satisfied (outside window, ineligible, or already elected).

  2. Election-complete path: a active-tagged affidavit key-pair exists, election constraints are satisfied, and the elect extrinsic is successfully submitted.

In all other cases, the routine retries by re-entering the affidavit key initialization phase to re-validate offchain storage and keystore consistency.

All behavior is supplied by trait implementations operating on this type.

§FlowChart

Fields§

§by: T::Public

Public key authorized to sign election transactions.

§at: BlockNumberFor<T>

Block number at which election execution is initiated.

Trait Implementations§

Source§

impl<T: Clone + Config> Clone for TryElection<T>
where T::Public: Clone,

Source§

fn clone(&self) -> TryElection<T>

Returns a duplicate of the value. Read more
1.0.0 · Source§

fn clone_from(&mut self, source: &Self)

Performs copy-assignment from source. Read more
Source§

impl<T> Debug for TryElection<T>
where T: Debug + Config,

Source§

fn fmt(&self, fmt: &mut Formatter<'_>) -> Result

Formats the value using the given formatter. Read more
Source§

impl<T: Config> Decode for TryElection<T>
where T::Public: Decode, BlockNumberFor<T>: Decode,

Source§

fn decode<__CodecInputEdqy: Input>( __codec_input_edqy: &mut __CodecInputEdqy, ) -> Result<Self, Error>

Attempt to deserialise the value from input.
§

fn decode_into<I>( input: &mut I, dst: &mut MaybeUninit<Self>, ) -> Result<DecodeFinished, Error>
where I: Input,

Attempt to deserialize the value from input into a pre-allocated piece of memory. Read more
§

fn skip<I>(input: &mut I) -> Result<(), Error>
where I: Input,

Attempt to skip the encoded value from input. Read more
§

fn encoded_fixed_size() -> Option<usize>

Returns the fixed encoded size of the type. Read more
Source§

impl<T: Config> Encode for TryElection<T>
where T::Public: Encode, BlockNumberFor<T>: Encode,

Source§

fn size_hint(&self) -> usize

If possible give a hint of expected size of the encoding. Read more
Source§

fn encode_to<__CodecOutputEdqy: Output + ?Sized>( &self, __codec_dest_edqy: &mut __CodecOutputEdqy, )

Convert self to a slice and append it to the destination.
§

fn encode(&self) -> Vec<u8>

Convert self to an owned vector.
§

fn using_encoded<R, F>(&self, f: F) -> R
where F: FnOnce(&[u8]) -> R,

Convert self to a slice and then invoke the given closure with it.
§

fn encoded_size(&self) -> usize

Calculates the encoded size. Read more
Source§

impl<T: Config> MaxEncodedLen for TryElection<T>
where T::Public: MaxEncodedLen, BlockNumberFor<T>: MaxEncodedLen,

Source§

fn max_encoded_len() -> usize

Upper bound, in bytes, of the maximum encoded size of this item.
Source§

impl<T: PartialEq + Config> PartialEq for TryElection<T>
where T::Public: PartialEq,

Source§

fn eq(&self, other: &TryElection<T>) -> bool

Tests for self and other values to be equal, and is used by ==.
1.0.0 · Source§

fn ne(&self, other: &Rhs) -> bool

Tests for !=. The default implementation is almost always sufficient, and should not be overridden without very good reason.
Source§

impl<T: Config> RoutineOf<<T as SigningTypes>::Public, <<<T as Config>::Block as HeaderProvider>::HeaderT as Header>::Number> for TryElection<T>

Authorization layer for the election execution routine.

This implementation resolves the affidavit public key authorized to run the election flow. It deliberately reuses the key-resolution logic from DeclareAffidavit to enforce strict lifecycle ordering:

Affidavit Declaration -> Key Rotation -> Election

By delegating authorization instead of duplicating it, this layer guarantees that elections are executed only by authors who have successfully completed the affidavit and key-rotation process.

Source§

fn who(at: &BlockNumberFor<T>) -> Result<T::Public, Self::Logger>

Determines the affidavit public key authorized to run the election routine.

§Semantics
  • Reuses the currently active affidavit key resolved by DeclareAffidavit.
  • Ensures that election execution is authorized by the same operational key that is eligible for affidavit declaration.
§Rationale

Elections are permitted only after a successful affidavit declaration and key-rotation cycle. By delegating to DeclareAffidavit::who, this routine enforces a strict lifecycle ordering without duplicating key-resolution logic.

§Key Resolution Note

Although this appears syntactically as the active affidavit key, it semantically represents the recently rotated next affidavit key that was promoted to active status by the previous OCW execution’s final routine (RotateAffidavitKey).

Source§

impl<T: Config> Routines<<<<T as Config>::Block as HeaderProvider>::HeaderT as Header>::Number> for TryElection<T>

Offchain routine responsible for opportunistic election execution.

This routine attempts to submit the elect extrinsic for the upcoming session using the currently active affidavit key.

It is intentionally designed to be:

  • Non-blocking: exits silently when prerequisites are unmet.
  • Idempotent: safe to run repeatedly across blocks and forks.
  • Pre-emptive: elections may be attempted before the OCW pipeline fully converges, succeeding in later executions.

The routine participates in the OCW execution pipeline:

InitAffidavitKey -> TryElection -> DeclareAffidavit -> RotateAffidavitKey

and relies on repeated OCW invocations to eventually satisfy all temporal and state-dependent constraints.

Source§

fn can_run(&self) -> Result<(), Self::Logger>

Checks whether an election can be processed in the current block.

§Semantics
§Behavior
  • If the election window is not open, the routine exits early with an informational log.
  • Hard failures (e.g. invalid configuration) are surfaced as logged errors.

This design allows the OCW orchestrator to continue executing subsequent routines (e.g. affidavit declaration) in the same block.

Source§

fn run_service(&self) -> Result<(), Self::Logger>

Attempts to submit the election transaction for the upcoming session.

§Execution Strategy

This routine is opportunistic and non-blocking:

  • If the election window is not open, execution passes silently to next routine.
  • If the author is not eligible, execution exits silently.
  • If the author already acted as the election runner, execution exits silently.

This behavior is intentional and allows the OCW pipeline to be orchestrated in the following order:

InitAffidavitKey -> TryElection -> DeclareAffidavit -> RotateAffidavitKey

Elections are therefore attempted pre-emptively and may succeed in a later OCW invocation once all prerequisites converge.

§Semantics
  • Uses the currently active affidavit key to authorize the election.
  • Submits an unsigned elect extrinsic signed offchain.
  • Ensures that each author runs the election at most once per session.
§Failure Handling
  • All signing or submission failures are logged as errors.
  • No retries are attempted in the same block.
  • Subsequent OCW executions may retry automatically.
Source§

fn on_ran_service(&self)

Logs a info message on a successful TryElection routine.

Source§

impl<T> TypeInfo for TryElection<T>
where T::Public: TypeInfo + 'static, BlockNumberFor<T>: TypeInfo + 'static, T: Config + 'static,

Source§

type Identity = TryElection<T>

The type identifying for which type info is provided. Read more
Source§

fn type_info() -> Type

Returns the static type identifier for Self.
Source§

impl<T: Config> EncodeLike for TryElection<T>
where T::Public: Encode, BlockNumberFor<T>: Encode,

Source§

impl<T: Eq + Config> Eq for TryElection<T>
where T::Public: Eq,

Source§

impl<T: Config> StructuralPartialEq for TryElection<T>

Auto Trait Implementations§

§

impl<T> Freeze for TryElection<T>
where <T as SigningTypes>::Public: Freeze, <<<T as Config>::Block as Block>::Header as Header>::Number: Freeze,

§

impl<T> RefUnwindSafe for TryElection<T>
where <T as SigningTypes>::Public: RefUnwindSafe, <<<T as Config>::Block as Block>::Header as Header>::Number: RefUnwindSafe,

§

impl<T> Send for TryElection<T>
where <T as SigningTypes>::Public: Send,

§

impl<T> Sync for TryElection<T>
where <T as SigningTypes>::Public: Sync,

§

impl<T> Unpin for TryElection<T>
where <T as SigningTypes>::Public: Unpin, <<<T as Config>::Block as Block>::Header as Header>::Number: Unpin,

§

impl<T> UnwindSafe for TryElection<T>
where <T as SigningTypes>::Public: UnwindSafe, <<<T as Config>::Block as Block>::Header as Header>::Number: UnwindSafe,

Blanket Implementations§

Source§

impl<T> Any for T
where T: 'static + ?Sized,

Source§

fn type_id(&self) -> TypeId

Gets the TypeId of self. Read more
Source§

impl<T> Borrow<T> for T
where T: ?Sized,

Source§

fn borrow(&self) -> &T

Immutably borrows from an owned value. Read more
Source§

impl<T> BorrowMut<T> for T
where T: ?Sized,

Source§

fn borrow_mut(&mut self) -> &mut T

Mutably borrows from an owned value. Read more
§

impl<T> CheckedConversion for T

§

fn checked_from<T>(t: T) -> Option<Self>
where Self: TryFrom<T>,

Convert from a value of T into an equivalent instance of Option<Self>. Read more
§

fn checked_into<T>(self) -> Option<T>
where Self: TryInto<T>,

Consume self to return Some equivalent value of Option<T>. Read more
Source§

impl<T> CloneToUninit for T
where T: Clone,

Source§

unsafe fn clone_to_uninit(&self, dest: *mut u8)

🔬This is a nightly-only experimental API. (clone_to_uninit)
Performs copy-assignment from self to dest. Read more
§

impl<T> DecodeAll for T
where T: Decode,

§

fn decode_all(input: &mut &[u8]) -> Result<T, Error>

Decode Self and consume all of the given input data. Read more
§

impl<T> DecodeLimit for T
where T: Decode,

§

fn decode_all_with_depth_limit( limit: u32, input: &mut &[u8], ) -> Result<T, Error>

Decode Self and consume all of the given input data. Read more
§

fn decode_with_depth_limit<I>(limit: u32, input: &mut I) -> Result<T, Error>
where I: Input,

Decode Self with the given maximum recursion depth and advance input by the number of bytes consumed. Read more
Source§

impl<T> DynClone for T
where T: Clone,

§

impl<Q, K> Equivalent<K> for Q
where Q: Eq + ?Sized, K: Borrow<Q> + ?Sized,

§

fn equivalent(&self, key: &K) -> bool

Checks if this value is equivalent to the given key. Read more
§

impl<Q, K> Equivalent<K> for Q
where Q: Eq + ?Sized, K: Borrow<Q> + ?Sized,

§

fn equivalent(&self, key: &K) -> bool

Checks if this value is equivalent to the given key. Read more
Source§

impl<T> From<T> for T

Source§

fn from(t: T) -> T

Returns the argument unchanged.

§

impl<T> Hashable for T
where T: Codec,

§

fn blake2_128(&self) -> [u8; 16]

§

fn blake2_256(&self) -> [u8; 32]

§

fn blake2_128_concat(&self) -> Vec<u8>

§

fn twox_128(&self) -> [u8; 16]

§

fn twox_256(&self) -> [u8; 32]

§

fn twox_64_concat(&self) -> Vec<u8>

§

fn identity(&self) -> Vec<u8>

§

impl<T> Instrument for T

§

fn instrument(self, span: Span) -> Instrumented<Self>

Instruments this type with the provided [Span], returning an Instrumented wrapper. Read more
§

fn in_current_span(self) -> Instrumented<Self>

Instruments this type with the current Span, returning an Instrumented wrapper. Read more
Source§

impl<T, U> Into<U> for T
where U: From<T>,

Source§

fn into(self) -> U

Calls U::from(self).

That is, this conversion is whatever the implementation of From<T> for U chooses to do.

Source§

impl<T> IntoEither for T

Source§

fn into_either(self, into_left: bool) -> Either<Self, Self>

Converts self into a Left variant of Either<Self, Self> if into_left is true. Converts self into a Right variant of Either<Self, Self> otherwise. Read more
Source§

fn into_either_with<F>(self, into_left: F) -> Either<Self, Self>
where F: FnOnce(&Self) -> bool,

Converts self into a Left variant of Either<Self, Self> if into_left(&self) returns true. Converts self into a Right variant of Either<Self, Self> otherwise. Read more
§

impl<T, U> IntoKey<U> for T
where U: FromKey<T>,

§

fn into_key(self) -> U

Source§

impl<T, U, Tag> IntoTag<U, Tag> for T
where U: FromTag<T, Tag>, Tag: DiscriminantTag,

Source§

fn into_tag(self) -> U

§

impl<Src, Dest> IntoTuple<Dest> for Src
where Dest: FromTuple<Src>,

§

fn into_tuple(self) -> Dest

§

impl<T> IsType<T> for T

§

fn from_ref(t: &T) -> &T

Cast reference.
§

fn into_ref(&self) -> &T

Cast reference.
§

fn from_mut(t: &mut T) -> &mut T

Cast mutable reference.
§

fn into_mut(&mut self) -> &mut T

Cast mutable reference.
§

impl<T, Outer> IsWrappedBy<Outer> for T
where Outer: AsRef<T> + AsMut<T> + From<T>, T: From<Outer>,

§

fn from_ref(outer: &Outer) -> &T

Get a reference to the inner from the outer.

§

fn from_mut(outer: &mut Outer) -> &mut T

Get a mutable reference to the inner from the outer.

§

impl<T> KeyedVec for T
where T: Codec,

§

fn to_keyed_vec(&self, prepend_key: &[u8]) -> Vec<u8>

Return an encoding of Self prepended by given slice.
Source§

impl<T, Time> Logging<Time> for T
where Time: Time,

Source§

const FALLBACK_TARGET: &'static str = "routine"

Default logging target if none is provided.

Most routines, especially offchain workers or background tasks, use this target for simplicity.

It allows a consistent place to look for routine logs without requiring every call to specify a target.

Note: This target is only a conveninence and may be somewhat vague. To ensure errors can still be traced accurately, the logged messages should include additional metadata (e.g., module name, error index, or contextual info) so that the source of the error can be identified even if the target is generic.

Source§

type Logger = DispatchError

The type taken and returned for logging.

We simply return the same [DispatchError] that was logged, so logging does not change control flow or error propagation.

DispatchError is used because in Substrate it encompasses all runtime errors - including module errors, token errors, arithmetic issues, and transactional boundaries - making it the universal substrate-side error representation.

Source§

type Level = LogLevel

The log level type.

We use the LogLevel enum to standardize severity levels (Info, Warn, Error, Debug) across all routine logs.

Source§

fn log( level: <T as Logging<Time>>::Level, err: &<T as Logging<Time>>::Logger, timestamp: Time, target: Option<&str>, fmt: Option<fn(Time, &<T as Logging<Time>>::Level, &str, &str) -> String>, ) -> <T as Logging<Time>>::Logger

Core logging function that all helpers delegate to. Read more
Source§

fn info( err: &Self::Logger, timestamp: Timestamp, target: Option<&str>, fmt: Option<fn(Timestamp, &Self::Level, &str, &str) -> String>, ) -> Self::Logger
where Self: Sized,

Logs an info-level message. Read more
Source§

fn warn( err: &Self::Logger, timestamp: Timestamp, target: Option<&str>, fmt: Option<fn(Timestamp, &Self::Level, &str, &str) -> String>, ) -> Self::Logger
where Self: Sized,

Logs a warning-level message. Read more
Source§

fn error( err: &Self::Logger, timestamp: Timestamp, target: Option<&str>, fmt: Option<fn(Timestamp, &Self::Level, &str, &str) -> String>, ) -> Self::Logger
where Self: Sized,

Logs an error-level message. Read more
Source§

fn debug( err: &Self::Logger, timestamp: Timestamp, target: Option<&str>, fmt: Option<fn(Timestamp, &Self::Level, &str, &str) -> String>, ) -> Self::Logger
where Self: Sized,

Logs a debug-level message. Read more
Source§

impl<T> Same for T

Source§

type Output = T

Should always be Self
§

impl<T> SaturatedConversion for T

§

fn saturated_from<T>(t: T) -> Self
where Self: UniqueSaturatedFrom<T>,

Convert from a value of T into an equivalent instance of Self. Read more
§

fn saturated_into<T>(self) -> T
where Self: UniqueSaturatedInto<T>,

Consume self to return an equivalent value of T. Read more
Source§

impl<T> ToOwned for T
where T: Clone,

Source§

type Owned = T

The resulting type after obtaining ownership.
Source§

fn to_owned(&self) -> T

Creates owned data from borrowed data, usually by cloning. Read more
Source§

fn clone_into(&self, target: &mut T)

Uses borrowed data to replace owned data, usually by cloning. Read more
Source§

impl<T, U> TryFrom<U> for T
where U: Into<T>,

Source§

type Error = Infallible

The type returned in the event of a conversion error.
Source§

fn try_from(value: U) -> Result<T, <T as TryFrom<U>>::Error>

Performs the conversion.
Source§

impl<T, U> TryInto<U> for T
where U: TryFrom<T>,

Source§

type Error = <U as TryFrom<T>>::Error

The type returned in the event of a conversion error.
Source§

fn try_into(self) -> Result<U, <U as TryFrom<T>>::Error>

Performs the conversion.
§

impl<T, U> TryIntoKey<U> for T
where U: TryFromKey<T>,

§

type Error = <U as TryFromKey<T>>::Error

§

fn try_into_key(self) -> Result<U, <U as TryFromKey<T>>::Error>

§

impl<S, T> UncheckedInto<T> for S
where T: UncheckedFrom<S>,

§

fn unchecked_into(self) -> T

The counterpart to unchecked_from.
§

impl<T, S> UniqueSaturatedInto<T> for S
where T: Bounded, S: TryInto<T>,

§

fn unique_saturated_into(self) -> T

Consume self to return an equivalent value of T.
§

impl<V, T> VZip<V> for T
where V: MultiLane<T>,

§

fn vzip(self) -> V

§

impl<T> WithSubscriber for T

§

fn with_subscriber<S>(self, subscriber: S) -> WithDispatch<Self>
where S: Into<Dispatch>,

Attaches the provided Subscriber to this type, returning a [WithDispatch] wrapper. Read more
§

fn with_current_subscriber(self) -> WithDispatch<Self>

Attaches the current default Subscriber to this type, returning a [WithDispatch] wrapper. Read more
§

impl<S> Codec for S
where S: Decode + Encode,

Source§

impl<Provider, Discriminant, T> DelegateVirtualDynBound<Provider, Discriminant> for T
where Provider: VirtualDynBound<Discriminant>, Discriminant: DiscriminantTag,

Source§

impl<Provider, Discriminant, T> DelegateVirtualStaticBound<Provider, Discriminant> for T
where Provider: VirtualStaticBound<Discriminant>, Discriminant: DiscriminantTag,

Source§

impl<T> Elastic for T
where T: RuntimeType + MaxEncodedLen,

§

impl<T> EncodeLike<&&T> for T
where T: Encode,

§

impl<T> EncodeLike<&T> for T
where T: Encode,

§

impl<T> EncodeLike<&mut T> for T
where T: Encode,

§

impl<T> EncodeLike<Arc<T>> for T
where T: Encode,

§

impl<T> EncodeLike<Box<T>> for T
where T: Encode,

§

impl<T> EncodeLike<Cow<'_, T>> for T
where T: ToOwned + Encode,

§

impl<T> EncodeLike<Rc<T>> for T
where T: Encode,

§

impl<S> FullCodec for S
where S: Decode + FullEncode,

§

impl<S> FullEncode for S
where S: Encode + EncodeLike,

§

impl<T> JsonSchemaMaybe for T

§

impl<T> MaybeDebug for T
where T: Debug,

§

impl<T> MaybeRefUnwindSafe for T
where T: RefUnwindSafe,

§

impl<T> Member for T
where T: Send + Sync + Debug + Eq + PartialEq + Clone + 'static,

Source§

impl<T> MetaAware for T
where T: TypeInfo + Debug,

Source§

impl<T> Portable for T
where T: Encode + Decode + Clone,

Source§

impl<T> Probe for T
where T: Encode + Debug + ?Sized,

Source§

impl<T> RuntimeEnum for T
where T: FullCodec + TypeInfo + Debug,

Source§

impl<T> RuntimeType for T
where T: Portable + MetaAware + Eq,

§

impl<T> StaticTypeInfo for T
where T: TypeInfo + 'static,

Source§

impl<T> Storable for T
where T: RuntimeType + 'static + EncodeLike,