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
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
//!# Asset_Management Pallet
//!
//!The Asset_Management pallet is used for everything that is related to
//!the asset management by its owners.
//!
//!## Overview
//!
//!The Asset_Management gives the possibility to the asset Owners, through governance,
//!to implement the following actions:
//! - Elect a Representative that will micro-manage the asset
//! - Demote a previously elected Representative
//! - Allow the representative to submit a list of Tenants to the Owners
//! - Allow the owners to vote on list of tenants submitted by the Representative
//!
//!### Dispatchable Functions
//!
//! * `launch_representative_session` - An Owner creates a referendum for the following available
//!   proposals:
//!   - Elect a Representative.
//!   - Demote a Representative.
//!
//! * `owners_vote` - Each asset owner can vote in an ongoing referendum.
//!
//! * `request_asset_management` - An active Representative can request an additional asset to manage.
//!
//! * `representative_approval` - Call used as a proposal for Representative election.
//!
//! * `demote_representative` - Call used as a proposal for Representative demotion.
//!
//! * `launch_tenant_session` - A Representative creates a referendum for the following available
//!   proposals:
//!   - Admit a Tenant for a given asset.
//!   - Evict  a Tenant from a given asset.
//!   The Representative has to submit a judgement about the tenant profile. This judgement
//!   will be considered by the owners before voting.
//!   Representatives receive a judgement fee from the aspiring tenant.
//!   A positive result of the referendum will send a guaranty_deposit payment request to the
//! tenant.   When the tenant finally pays the guaranty_deposit,his account is connected to the
//!   asset through `link_tenant_to_asset` and this marks the start of his contract with the owners.
//!
//! * `link_tenant_to_asset` - Call used as a proposal to link an accepted tenant with an existing
//!   asset.
//!
//! * `unlink_tenant_to_asset` - Call used as a proposal to remove the link between a tenant and an
//!   asset.
//!
//! * `request_guaranty_payment` - Call used to send a guaranty deposit payment request to a tenant.

#![cfg_attr(not(feature = "std"), no_std)]

pub use pallet::*;
pub use pallet_assets as Assetss;
pub use pallet_democracy as Dem;
pub use pallet_housing_fund as HFund;
pub use pallet_identity as Ident;
pub use pallet_nft as Nft;
pub use pallet_onboarding as Onboarding;
pub use pallet_payment as Payment;
pub use pallet_roles as Roles;
pub use pallet_share_distributor as Share;

mod functions;
mod types;
pub use crate::types::*;
pub use functions::*;

#[cfg(test)]
mod mock;

#[cfg(test)]
mod tests;

//#[cfg(feature = "runtime-benchmarks")]
//mod benchmarking;

#[frame_support::pallet]
pub mod pallet {
	use super::*;
	use frame_system::WeightInfo;

	#[pallet::pallet]
	#[pallet::generate_store(pub(super) trait Store)]
	#[pallet::without_storage_info]
	pub struct Pallet<T>(_);

	/// Configure the pallet by specifying the parameters and types on which it depends.
	#[pallet::config]
	pub trait Config:
		frame_system::Config
		+ HFund::Config
		+ Onboarding::Config
		+ Roles::Config
		+ Dem::Config
		+ Share::Config
		+ Nft::Config
		+ Assetss::Config
		+ Ident::Config
		+ Payment::Config
	{
		/// Because this pallet emits events, it depends on the runtime's definition of an event.
		type Event: From<Event<Self>> + IsType<<Self as frame_system::Config>::Event>;
		type Call: Parameter
			+ UnfilteredDispatchable<Origin = <Self as frame_system::Config>::Origin>
			+ From<Call<Self>>
			+ GetDispatchInfo;
		type Delay: Get<Self::BlockNumber>;
		type CheckDelay: Get<Self::BlockNumber>;
		type InvestorVoteAmount: Get<u128>;
		type Currency: Currency<Self::AccountId> + ReservableCurrency<Self::AccountId>;
		type WeightInfo: WeightInfo;

		/// Number of months for the guaranty deposit
		#[pallet::constant]
		type Guaranty: Get<u32>;

		/// Return on Rent
		#[pallet::constant]
		type RoR: Get<Percent>;

		#[pallet::constant]
		type MinimumDepositVote: Get<BalanceOf<Self>>;

		/// Fees payed to the Representative by the tenant, to provide a judgement
		#[pallet::constant]
		type RepFees: Get<BalanceOf<Self>>;

		/// Lease period in number of blocks
		#[pallet::constant]
		type ContractLength: Get<Self::BlockNumber>;

		/// Period between check of Referendum status
		#[pallet::constant]
		type CheckPeriod: Get<Self::BlockNumber>;

		/// Period between check of rent payment status for active tenants
		#[pallet::constant]
		type RentCheck: Get<Self::BlockNumber>;

		/// Lease period in number of months
		#[pallet::constant]
		type Lease: Get<u32>;

		/// Maintenance fees percentage taken on monthly rent
		#[pallet::constant]
		type Maintenance: Get<Percent>;
	}

	//Store the referendum_index and the struct containing the
	// virtual_account/caller/potential_rep/vote_result
	#[pallet::storage]
	#[pallet::getter(fn proposals)]
	pub type ProposalsLog<T: Config> =
		StorageMap<_, Blake2_128Concat, Dem::ReferendumIndex, ProposalRecord<T>, OptionQuery>;

	#[pallet::storage]
	#[pallet::getter(fn guaranty)]
	pub type GuarantyPayment<T: Config> = StorageDoubleMap<
		_,
		Blake2_128Concat,
		T::AccountId, // payment issuer
		Blake2_128Concat,
		T::AccountId, // payment recipient
		Payment::PaymentDetail<T>,
	>;

	#[pallet::storage]
	#[pallet::getter(fn indexes)]
	pub type ProposalsIndexes<T: Config> =
		StorageMap<_, Blake2_128Concat, T::AccountId, Dem::ReferendumIndex, OptionQuery>;

	// Pallets use events to inform users when important changes are made.
	// https://docs.substrate.io/main-docs/build/events-errors/
	#[pallet::event]
	#[pallet::generate_deposit(pub(super) fn deposit_event)]
	pub enum Event<T: Config> {
		///A voting session to elect a representative has started
		RepresentativeVoteSessionStarted {
			caller: T::AccountId,
			candidate: T::AccountId,
			asset_account: T::AccountId,
		},
		/// A voting session to link a tenant to an asset has started
		TenantVoteSessionStarted {
			representative: T::AccountId,
			tenant: T::AccountId,
			asset_account: T::AccountId,
		},
		///An investor voted
		InvestorVoted {
			caller: T::AccountId,
			session_number: Dem::ReferendumIndex,
			when: BlockNumberOf<T>,
		},
		///A representative role was granted
		RepresentativeCandidateApproved {
			candidate: T::AccountId,
			asset_account: T::AccountId,
			when: BlockNumberOf<T>,
		},
		///An account was stripped of its Representative role
		RepresentativeDemoted {
			candidate: T::AccountId,
			asset_account: T::AccountId,
			when: BlockNumberOf<T>,
		},
		/// A tenant is linked with an asset
		TenantLinkedToAsset {
			tenant: T::AccountId,
			collection: T::NftCollectionId,
			item: T::NftItemId,
			asset_account: T::AccountId,
		},
		/// A tenant is demoted and unlinked with an asset
		TenantDemoted {
			tenant: T::AccountId,
			collection: T::NftCollectionId,
			item: T::NftItemId,
			asset_account: T::AccountId,
		},
		///The amount of the tenant debt
		TenantDebt { tenant: T::AccountId, debt: BalanceOf<T>, when: BlockNumberOf<T> },

		///Rent distributed to owners
		RentDistributed { owners: Vec<T::AccountId>, amount: BalanceOf<T>, when: BlockNumberOf<T> },

		/// Guaranty payment request sent
		GuarantyPaymentRequested {
			tenant: T::AccountId,
			asset_account: T::AccountId,
			amount: Payment::BalanceOf<T>,
			when: BlockNumberOf<T>,
		},

		//Asset maintenance fees, have been taken from the rent received, and reserved
		MaintenanceFeesPayment {
			tenant: T::AccountId,
			when: BlockNumberOf<T>,
			asset_account: T::AccountId,
			amount: BalanceOf<T>,
		},
	}

	// Errors inform users that something went wrong.
	#[pallet::error]
	pub enum Error<T> {
		/// Error names should be descriptive.
		NoneValue,
		/// The account is not an Asset account
		NotAnAssetAccount,
		/// The account is not in the representative waiting list
		NotAPendingRepresentative,
		/// The account is not a representative
		NotARepresentative,
		/// Not an active Representative
		NotAnActiveRepresentative,
		/// The asset is already linked with a representative
		AssetAlreadyLinkedWithRepresentative,
		/// The asset is not linked with a representative
		AssetNotLinkedWithRepresentative,
		/// The given representative is not linked with the asset
		InvalidRepresentative,
		/// The asset is not linked to the representative
		AssetOutOfControl,
		/// The candidate is not a tenant
		NotATenant,
		/// An asset is already linked with the representative
		RepresentativeAlreadyLinkedWithAsset,
		/// An asset is already linked with the tenant
		TenantAlreadyLinkedWithAsset,
		/// The tenant is not linked to the asset
		TenantAssetNotLinked,
		/// Errors should have helpful documentation associated with them.
		StorageOverflow,
		/// The proposal could not be created
		FailedToCreateProposal,
		/// Not an owner in the corresponding virtual account
		NotAnOwner,
		/// The Asset Does not Exists
		NotAnAsset,
		/// This referendum does not exists
		NotAValidReferendum,
		/// This referendum is over
		ReferendumCompleted,
		/// Not enough funds in the account
		NotEnoughFunds,
		/// Payment request already sent
		ExistingPaymentRequest,
		/// Not enough funds in the tenant account
		NotEnoughTenantFunds,
		/// The Tenant did not provide detailed information
		NotARegisteredTenant,
		/// Existing Representative request
		ExistingPendingRequest,
		/// Maximum number of tenants reached
		MaximumNumberOfTenantsReached,
	}

	#[pallet::hooks]
	impl<T: Config> Hooks<BlockNumberFor<T>> for Pallet<T> {
		fn on_initialize(n: T::BlockNumber) -> Weight {
			Self::begin_block(n)
		}

		fn on_idle(n: T::BlockNumber, _max_weight: Weight) -> Weight {
			Self::finish_block(n)
		}
	}

	#[pallet::call]
	impl<T: Config> Pallet<T> {
		/// Build the call to be executed when the proposal pass the democracy vote
		/// The origin must but root
		/// - account_id : the virtual account of the asset of the proposal
		/// - proposal : call encapsulating the inital proposal
		#[pallet::weight(10_000 + T::DbWeight::get().reads_writes(1,1).ref_time())]
		pub fn execute_call_dispatch(
			origin: OriginFor<T>,
			account_id: AccountIdOf<T>,
			proposal: Box<<T as Config>::Call>,
		) -> DispatchResultWithPostInfo {
			ensure_root(origin)?;

			proposal
				.dispatch_bypass_filter(frame_system::RawOrigin::Signed(account_id.clone()).into())
				.ok();

			Ok(().into())
		}

		/// Using the function below, an active Representative can request an additional asset to manage.
		/// The origin must be an active Representative.
		/// - account_id: an account with the representative role
		#[pallet::weight(10_000 + T::DbWeight::get().reads_writes(1,1).ref_time())]
		pub fn request_asset_management(
			origin: OriginFor<T>,
			account_id: AccountIdOf<T>,
		) -> DispatchResultWithPostInfo {
			let caller = ensure_signed(origin.clone())?;
			if caller != account_id {
				ensure!(Roles::Pallet::<T>::servicers(&caller).is_some(), Roles::Error::<T>::OnlyForServicers);
			}
			let representative =
				<T as frame_system::Config>::Origin::from(RawOrigin::Signed(account_id.clone()));
			let rep_infos = Roles::Pallet::<T>::reps(caller).unwrap();
			
			//Caller is not already in Representative waiting list
			ensure!(
				!Roles::RepApprovalList::<T>::contains_key(&account_id),
				Error::<T>::ExistingPendingRequest
			);

			//Caller is a registered and activated Representative
			ensure!(
				Roles::RepresentativeLog::<T>::contains_key(&account_id),
				Error::<T>::NotAnActiveRepresentative
			);
			ensure!(rep_infos.activated,Error::<T>::NotAnActiveRepresentative);
			
			//Send request
			Roles::Representative::<T>::new(representative).ok();

			Ok(().into())
		}

		/// Using the function below, an owner triggers a vote session with a proposal for an asset
		/// The origin must be an owner of the asset
		/// - asset_type: type of the asset
		/// - asset_id: id of the asset
		/// - representative: an account with the representative role to be designed
		/// - proposal contains the extrinsics to be executed depending on the vote result
		#[pallet::weight(10_000 + T::DbWeight::get().reads_writes(1,1).ref_time())]
		pub fn launch_representative_session(
			origin: OriginFor<T>,
			asset_type: Nft::PossibleCollections,
			asset_id: T::NftItemId,
			representative: T::AccountId,
			proposal: VoteProposals,
		) -> DispatchResultWithPostInfo {
			let caller = ensure_signed(origin.clone())?;

			// Get asset virtual account if it exists
			let collection_id: T::NftCollectionId = asset_type.value().into();

			let ownership = Share::Pallet::<T>::virtual_acc(collection_id, asset_id);
			ensure!(ownership.is_some(), Error::<T>::NotAnAsset);
			let ownership = ownership.unwrap();

			let asset = Onboarding::Pallet::<T>::houses(collection_id, asset_id);
			ensure!(asset.is_some(), Error::<T>::NotAnAsset);
			let asset = asset.unwrap();

			// Ensure that the caller is one of the asset owners
			ensure!(ownership.owners.contains(&caller), Error::<T>::NotAnOwner);

			let virtual_account = ownership.virtual_account;

			// Create the call
			let proposal_call = match proposal {
				VoteProposals::Election => {
					// Check if the account is in the representative waiting list
					let rep = Roles::Pallet::<T>::get_pending_representatives(&representative);
					ensure!(rep.is_some(), Error::<T>::NotAPendingRepresentative);

					// Ensure that the asset doesn't have a representative yet
					ensure!(
						asset.representative.is_none(),
						Error::<T>::AssetAlreadyLinkedWithRepresentative
					);

					//Ensure that the Representative is not already connected to this asset
					ensure!(
						!rep.unwrap().assets_accounts.contains(&virtual_account),
						Error::<T>::RepresentativeAlreadyLinkedWithAsset
					);

					Call::<T>::representative_approval {
						rep_account: representative.clone(),
						collection: collection_id,
						item: asset_id,
					}
				},
				VoteProposals::Demotion => {
					// Ensure that the asset is linked with the representative
					let asset_rep = asset.representative;
					ensure!(asset_rep.is_some(), Error::<T>::AssetNotLinkedWithRepresentative);
					ensure!(
						asset_rep == Some(representative.clone()),
						Error::<T>::InvalidRepresentative
					);
					Call::<T>::demote_representative {
						rep_account: representative.clone(),
						collection: collection_id,
						item: asset_id,
					}
				},
			};

			let deposit = T::MinimumDeposit::get();
			//Ensure that the virtual account has enough funds
			for f in ownership.owners {
				<T as Dem::Config>::Currency::transfer(
					&f,
					&virtual_account,
					deposit,
					ExistenceRequirement::AllowDeath,
				)
				.ok();
			}

			//Format the call and create the proposal Hash
			let proposal_hash =
				Self::create_proposal_hash_and_note(virtual_account.clone(), proposal_call);

			let threshold = Dem::VoteThreshold::SimpleMajority;
			let delay = <T as Config>::Delay::get();

			let referendum_index =
				Dem::Pallet::<T>::internal_start_referendum(proposal_hash, threshold, delay);

			//Create data for proposals Log
			ProposalRecord::<T>::new(
				caller.clone(),
				virtual_account.clone(),
				representative.clone(),
				referendum_index,
				collection_id,
				asset_id,
			)
			.ok();

			//Emit Event
			Self::deposit_event(Event::RepresentativeVoteSessionStarted {
				caller,
				candidate: representative,
				asset_account: virtual_account,
			});

			Ok(().into())
		}

		/// The function below allows the owner to vote.
		/// The balance locked and used for vote conviction corresponds
		/// to the number of ownership tokens possessed by the voter.
		/// The origin must be an owner of the asset
		/// - referendum_index: index of the referendum the voter is taking part in
		/// - vote: aye or nay
		#[pallet::weight(10_000 + T::DbWeight::get().reads_writes(1,1).ref_time())]
		pub fn owners_vote(
			origin: OriginFor<T>,
			referendum_index: Dem::ReferendumIndex,
			vote: bool,
		) -> DispatchResult {
			let voter = ensure_signed(origin.clone())?;
			//Check that the referendum exists and is active
			ensure!(
				ProposalsLog::<T>::contains_key(referendum_index),
				Error::<T>::NotAValidReferendum
			);
			//Check the referendum status
			let infos = Self::proposals(referendum_index).unwrap();
			let status = infos.vote_result;
			ensure!(status == VoteResult::AWAITING, Error::<T>::ReferendumCompleted);
			//check that caller can vote
			let ownership =
				Share::Pallet::<T>::virtual_acc(infos.collection_id, infos.item_id).unwrap();
			ensure!(Self::caller_can_vote(&voter, ownership.clone()), Error::<T>::NotAnOwner);
			//Get number of FS tokens own by caller
			let tokens = Assetss::Pallet::<T>::balance(ownership.token_id.into(), &voter);
			let token0 = Self::assets_bal_to_u128(tokens).unwrap();
			let bals0 = BalanceType::<T>::convert_to_balance(token0);
			let token1 = bals0.dem_bal;

			//let v = Dem::Vote { aye: vote, conviction: Dem::Conviction::Locked1x };
			let v = Self::vote_helper(token0, vote).unwrap();
			Dem::Pallet::<T>::vote(
				origin.clone(),
				referendum_index,
				Dem::AccountVote::Standard { vote: v, balance: token1 },
			)
			.ok();

			//Emit event
			Self::deposit_event(Event::InvestorVoted {
				caller: voter,
				session_number: referendum_index,
				when: <frame_system::Pallet<T>>::block_number(),
			});

			Ok(())
		}

		/// The function below allows the approval of a Representative role request
		/// The origin must be the virtual account connected to the asset
		/// - rep_account: account Of the candidate to the representative account
		/// - collection: collection number of the asset.
		/// - item: item number of the asset.
		#[pallet::weight(10_000 + T::DbWeight::get().reads_writes(1,1).ref_time())]
		pub fn representative_approval(
			origin: OriginFor<T>,
			rep_account: T::AccountId,
			collection: T::NftCollectionId,
			item: T::NftItemId,
		) -> DispatchResult {
			let caller = ensure_signed(origin.clone())?;
			let asset_account =
				Share::Pallet::<T>::virtual_acc(collection, item).unwrap().virtual_account;

			//Check that the caller is a stored virtual account
			ensure!(caller == asset_account, Error::<T>::NotAnAssetAccount);

			Onboarding::Houses::<T>::mutate(collection, item, |asset| {
				let mut asset0 = asset.clone().unwrap();
				asset0.representative = Some(rep_account.clone());
				*asset = Some(asset0);
			});

			//Approve role request
			Self::approve_representative_role(origin, rep_account.clone()).ok();

			Self::deposit_event(Event::RepresentativeCandidateApproved {
				candidate: rep_account,
				asset_account: caller,
				when: <frame_system::Pallet<T>>::block_number(),
			});

			Ok(())
		}

		/// The function below allows the demotion of a previously elected Representative
		/// The origin must be the virtual account connected to the asset
		/// - rep_account: account Of the candidate to the representative account
		/// - collection: collection_id of the asset.
		/// - item: item_id of the asset.
		#[pallet::weight(10_000 + T::DbWeight::get().reads_writes(1,1).ref_time())]
		pub fn demote_representative(
			origin: OriginFor<T>,
			rep_account: T::AccountId,
			collection: T::NftCollectionId,
			item: T::NftItemId,
		) -> DispatchResult {
			let caller = ensure_signed(origin)?;

			//Check that the caller is a stored virtual account
			ensure!(
				caller
					== Share::Pallet::<T>::virtual_acc(collection, item).unwrap().virtual_account,
				Error::<T>::NotAnAssetAccount
			);

			//revoke Representative Role
			Self::revoke_representative_role(rep_account.clone()).ok();
			Onboarding::Houses::<T>::mutate(collection, item, |asset| {
				let mut asset0 = asset.clone().unwrap();
				asset0.representative = None;
				*asset = Some(asset0);
			});

			Self::deposit_event(Event::RepresentativeDemoted {
				candidate: rep_account,
				asset_account: caller,
				when: <frame_system::Pallet<T>>::block_number(),
			});

			Ok(())
		}

		/// Using the function below, a representative triggers a vote session with a proposal for a
		/// tenant to be linked with an asset The origin must be a representative
		/// - asset_type: type of the asset
		/// - asset_id: id of the asset
		/// - tenant: an account with the tenant role
		#[pallet::weight(10_000 + T::DbWeight::get().reads_writes(1,1).ref_time())]
		pub fn launch_tenant_session(
			origin: OriginFor<T>,
			asset_type: Nft::PossibleCollections,
			asset_id: T::NftItemId,
			tenant: T::AccountId,
			proposal: VoteProposals,
			judgement: Ident::Judgement<IdentBalanceOf<T>>,
		) -> DispatchResult {
			let caller = ensure_signed(origin.clone())?;

			// Ensure that the caller is a representative
			let rep = Roles::Pallet::<T>::reps(caller.clone());
			ensure!(rep.is_some(), Error::<T>::NotARepresentative);
			let rep = rep.unwrap();
			ensure!(rep.activated, Error::<T>::NotAnActiveRepresentative);

			// Get the asset virtual account if exists
			let collection_id: T::NftCollectionId = asset_type.value().into();
			let ownership = Share::Pallet::<T>::virtual_acc(collection_id, asset_id);
			ensure!(ownership.is_some(), Error::<T>::NotAnAsset);
			
			//Compare guaranty payment amount+fees with tenant free_balance
			let guaranty = Self::calculate_guaranty(collection_id, asset_id);
			let fee0 = Self::manage_bal_to_u128(T::RepFees::get()).unwrap();
			let bals0 = BalanceType::<T>::convert_to_balance(guaranty);
			let fee1 = T::IncentivePercentage::get() * bals0.manage_bal;
			let total_amount = guaranty + fee0 + Self::manage_bal_to_u128(fee1).unwrap();
			let tenant_bal0: BalanceOf<T> = <T as Config>::Currency::free_balance(&tenant);
			let tenant_bal = Self::manage_bal_to_u128(tenant_bal0).unwrap();

			let asset_account = ownership.unwrap().virtual_account;
			ensure!(rep.assets_accounts.contains(&asset_account), Error::<T>::AssetOutOfControl);

			// Ensure that provided account is a valid tenant
			let tenant0 = Roles::Pallet::<T>::tenants(tenant.clone());
			ensure!(tenant0.is_some(), Error::<T>::NotATenant);
			// Ensure that the tenant is registered
			let tenant_infos = Roles::Pallet::<T>::tenants(tenant.clone()).unwrap();
			ensure!(tenant_infos.registered, Error::<T>::NotARegisteredTenant);

			let tenant0 = tenant0.unwrap();
			match proposal {
				VoteProposals::Election => {
					// Ensure that the tenant is not linked to an asset
					ensure!(
						tenant0.asset_account.is_none(),
						Error::<T>::TenantAlreadyLinkedWithAsset
					);
					//Ensure there is no existing payment request for this asset
					ensure!(
						Self::guaranty(&tenant0.account_id, &asset_account).is_none(),
						Error::<T>::ExistingPaymentRequest
					);
					//ensure that tenant can pay Guaranty deposit
					ensure!(tenant_bal > total_amount, Error::<T>::NotEnoughTenantFunds);
					//provide judgement
					let index = rep.index;
					let target = T::Lookup::unlookup(tenant.clone());
					Ident::Pallet::<T>::provide_judgement(origin.clone(), index, target, judgement)
						.ok();
				},
				VoteProposals::Demotion => {
					// Ensure that the tenant is linked to the asset
					ensure!(
						tenant0.asset_account == Some(asset_account.clone()),
						Error::<T>::TenantAssetNotLinked
					);
					let house = Onboarding::Pallet::<T>::houses(collection_id, asset_id).unwrap();
					ensure!(house.tenants.contains(&tenant), Error::<T>::TenantAssetNotLinked)
				},
			};

			let deposit = T::MinimumDeposit::get();

			// Ensure that the representative has enough funds
			<T as Dem::Config>::Currency::transfer(
				&caller,
				&asset_account,
				deposit,
				ExistenceRequirement::AllowDeath,
			)
			.ok();

			let call = match proposal {
				VoteProposals::Election => Call::<T>::request_guaranty_payment {
					from: tenant.clone(),
					collection: collection_id,
					item: asset_id,
					judgement,
				},
				VoteProposals::Demotion => Call::<T>::unlink_tenant_to_asset {
					tenant: tenant.clone(),
					collection: collection_id,
					item: asset_id,
				},
			};

			let proposal_hash = Self::create_proposal_hash_and_note(asset_account.clone(), call);

			let threshold = Dem::VoteThreshold::SimpleMajority;

			let delay = <T as Config>::Delay::get();

			let referendum_index =
				Dem::Pallet::<T>::internal_start_referendum(proposal_hash, threshold, delay);

			// Create data for proposals Log
			ProposalRecord::<T>::new(
				caller.clone(),
				asset_account.clone(),
				tenant.clone(),
				referendum_index,
				collection_id,
				asset_id,
			)
			.ok();

			//Emit Event
			Self::deposit_event(Event::TenantVoteSessionStarted {
				representative: caller,
				tenant,
				asset_account,
			});

			Ok(())
		}

		/// The function below links an accepted tenant with an existing asset
		/// The origin must be the virtual account connected to the asset
		/// - tenant: an account with the tenant role
		/// - collection: collection_id of the asset
		/// - item: item_id of the asset
		#[pallet::weight(10_000 + T::DbWeight::get().reads_writes(1,1).ref_time())]
		pub fn link_tenant_to_asset(
			origin: OriginFor<T>,
			tenant: T::AccountId,
			collection: T::NftCollectionId,
			item: T::NftItemId,
		) -> DispatchResult {
			let caller = ensure_signed(origin)?;

			// Ensure the caller is the virtual account of the asset
			let asset_account =
				Share::Pallet::<T>::virtual_acc(collection, item).unwrap().virtual_account;
			ensure!(caller == asset_account, Error::<T>::NotAnAssetAccount);

			Self::tenant_link_asset(tenant.clone(), collection, item, asset_account.clone()).ok();

			Self::deposit_event(Event::TenantLinkedToAsset {
				tenant,
				collection,
				item,
				asset_account,
			});

			Ok(())
		}

		/// The function below sends a guaranty deposiy payment request to a tenant. This extrinsic
		/// is executed After a positive tenant_session.
		/// The origin must be the virtual account connected to the asset
		/// - tenant: an account with the tenant role linked to the asset
		/// - collection: collection_id of the asset
		/// - item: item_id of the asset
		/// - _judgement is provided by the representative while creating a tenant session
		#[pallet::weight(10_000 + T::DbWeight::get().reads_writes(1,1).ref_time())]
		pub fn request_guaranty_payment(
			origin: OriginFor<T>,
			from: T::AccountId,
			collection: T::NftCollectionId,
			item: T::NftItemId,
			_judgement: Ident::Judgement<IdentBalanceOf<T>>,
		) -> DispatchResult {
			let creator = ensure_signed(origin.clone())?;

			// Ensure the caller is the virtual account of the asset
			let asset_account =
				Share::Pallet::<T>::virtual_acc(collection, item).unwrap().virtual_account;
			ensure!(creator == asset_account, Error::<T>::NotAnAssetAccount);

			// Check vacancy state of the asset
			let vacancy = Self::fetch_house(collection,item).max_tenants;
			ensure!(vacancy > 0, Error::<T>::MaximumNumberOfTenantsReached);

			// Check for awaiting guaranty payment requests
			let requests = Payment::Payment::<T>::iter().count();
			ensure!(vacancy > requests as u8, Error::<T>::MaximumNumberOfTenantsReached);
			
			//Launch payment request
			Self::guaranty_payment(origin, from.clone(), collection, item).ok();
			let payment = Self::guaranty(from.clone(), asset_account).unwrap();
			let now = <frame_system::Pallet<T>>::block_number();

			Self::deposit_event(Event::GuarantyPaymentRequested {
				tenant: from,
				asset_account: creator,
				amount: payment.amount,
				when: now,
			});

			Ok(())
		}

		/// The function below unlinks a tenant with an asset
		/// The origin must be the virtual account connected to the asset
		/// - tenant: an account with the tenant role linked to the asset
		/// - collection: collection_id of the asset
		/// - item: item_id of the asset
		#[pallet::weight(10_000 + T::DbWeight::get().reads_writes(1,1).ref_time())]
		pub fn unlink_tenant_to_asset(
			origin: OriginFor<T>,
			tenant: T::AccountId,
			collection: T::NftCollectionId,
			item: T::NftItemId,
		) -> DispatchResult {
			let caller = ensure_signed(origin)?;

			// Ensure the caller is the virtual account of the asset
			let asset_account =
				Share::Pallet::<T>::virtual_acc(collection, item).unwrap().virtual_account;
			ensure!(caller == asset_account, Error::<T>::NotAnAssetAccount);

			Self::tenant_unlink_asset(tenant.clone(), collection, item).ok();

			Self::deposit_event(Event::TenantDemoted { tenant, collection, item, asset_account });

			Ok(())
		}
	}
}