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
// This file is part of Substrate.

// Copyright (C) Parity Technologies (UK) Ltd.
// SPDX-License-Identifier: Apache-2.0

// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// 	http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.

//! Assets pallet's `StoredMap` implementation.

use super::*;

impl<T: Config<I>, I: 'static> StoredMap<(T::AssetId, T::AccountId), T::Extra> for Pallet<T, I> {
	fn get(id_who: &(T::AssetId, T::AccountId)) -> T::Extra {
		let (id, who) = id_who;
		Account::<T, I>::get(id, who).map(|a| a.extra).unwrap_or_default()
	}

	fn try_mutate_exists<R, E: From<DispatchError>>(
		id_who: &(T::AssetId, T::AccountId),
		f: impl FnOnce(&mut Option<T::Extra>) -> Result<R, E>,
	) -> Result<R, E> {
		let (id, who) = id_who;
		let mut maybe_extra = Account::<T, I>::get(id, who).map(|a| a.extra);
		let r = f(&mut maybe_extra)?;
		// They want to write some value or delete it.
		// If the account existed and they want to write a value, then we write.
		// If the account didn't exist and they want to delete it, then we let it pass.
		// Otherwise, we fail.
		Account::<T, I>::try_mutate(id, who, |maybe_account| {
			if let Some(extra) = maybe_extra {
				// They want to write a value. Let this happen only if the account actually exists.
				if let Some(ref mut account) = maybe_account {
					account.extra = extra;
				} else {
					return Err(DispatchError::NoProviders.into())
				}
			} else {
				// They want to delete it. Let this pass if the item never existed anyway.
				ensure!(maybe_account.is_none(), DispatchError::ConsumerRemaining);
			}
			Ok(r)
		})
	}
}