UNPKG

wowok_agent

Version:

Agent for WoWok: Unlock Co-Creation, Lighting Transaction, Empower Potential.

522 lines (487 loc) 25.4 kB
import { TransactionBlock, IsValidArgType, TxbAddress, TagName, PassportObject, Errors, ERROR, Permission, PermissionIndex, PermissionIndexType, BuyRequiredEnum, Customer_RequiredInfo, Service, Service_Guard_Percent, Service_Sale, Treasury, OrderResult, DicountDispatch as WowokDiscountDispatch, ProgressObject, Arbitration, Service_Discount, PermissionObject, } from 'wowok'; import { ObjectOrder, ObjectService, query_objects } from '../query/objects.js'; import { AccountOrMark_Address, CallBase, CallResult, GetAccountOrMark_Address, GetManyAccountOrMark_Address, GetObjectExisted, GetObjectMain, GetObjectParam, Namedbject, ObjectParam, ObjectTypedMain, ObjectsOp, TypeNamedObjectWithPermission, PayParam } from "./base.js"; import { Account } from '../local/account.js'; import { LocalMark } from '../local/local.js'; import { crypto_string } from '../common.js'; export interface ServiceWithdraw extends PayParam { withdraw_guard: string; } export interface DicountDispatch { receiver: AccountOrMark_Address; discount: Service_Discount; count?: number; } export interface RefundWithGuard { order:string; refund_guard:string; } export interface RefundWithArb { order:string; arb:string; } export type Service_Buy = { item: string; // max_price: string | number | bigint; count: string | number | bigint; } /// The execution priority is determined by the order in which the object attributes are arranged export interface CallService_Data { object: ObjectTypedMain; order_new?: {buy_items:Service_Buy[], discount_object?:string, customer_info_required?: string, namedNewOrder?: Namedbject, namedNewProgress?:Namedbject} order_agent?: {order?:string; agents: AccountOrMark_Address[];}; order_required_info?: {order:string; customer_info_required?:string}; order_refund?: RefundWithGuard | RefundWithArb; order_withdrawl?: {order:string; data:ServiceWithdraw}; // guard address order_payer?: {order?:string; payer_new:AccountOrMark_Address; }; // transfer the order payer permission to someaddress description?: string; endpoint?: string; payee_treasury?:ObjectParam; gen_discount?: DicountDispatch[]; repository?: ObjectsOp; extern_withdraw_treasury?: ObjectsOp; machine?: string; arbitration?: ObjectsOp; customer_required_info?: {pubkey:string; required_info:(string | BuyRequiredEnum)[]}; sales?: {op:'add', sales:Service_Sale[]} | {op:'remove'; sales_name:string[]} withdraw_guard?: {op:'add' | 'set'; guards:Service_Guard_Percent[]} | {op:'removeall'} | {op:'remove', guards:string[]}; refund_guard?: {op:'add' | 'set'; guards:Service_Guard_Percent[]} | {op:'removeall'} | {op:'remove', guards:string[]}; bPublished?: boolean; buy_guard?: string; bPaused?: boolean; clone_new?: {token_type_new?:string; namedNew?: Namedbject}; } export class CallService extends CallBase { data: CallService_Data; object_address: string | undefined = undefined; permission_address: string | undefined = undefined; type_parameter: string | undefined = undefined; constructor(data: CallService_Data) { super(); this.data = data; } protected async prepare(): Promise<void> { if (!this.object_address) { this.object_address = (await LocalMark.Instance().get_address(GetObjectExisted(this.data.object))); } if (this.object_address) { await this.update_content('Service', this.object_address); if (!this.content) ERROR(Errors.InvalidParam, 'CallService_Data.data.object:' + this.object_address); this.permission_address = (this.content as ObjectService).permission; this.type_parameter = Service.parseObjectType((this.content as ObjectService).type_raw); } else { const n = GetObjectMain(this.data.object) as TypeNamedObjectWithPermission; if (!IsValidArgType(n?.type_parameter)) { ERROR(Errors.IsValidArgType, 'CallService_Data.data.object.type_parameter'); } this.permission_address = (await LocalMark.Instance().get_address(GetObjectExisted(n?.permission))); this.type_parameter = (n as any)?.type_parameter; } } async call(account?:string) : Promise<CallResult> { var checkOwner = false; const guards : string[] = []; const perms : PermissionIndexType[] = []; await this.prepare(); if (this.permission_address) { if (!this.data?.object) { perms.push(PermissionIndex.service) } if (this.data?.description != null && this.object_address) { perms.push(PermissionIndex.service_description) } if (this.data?.bPaused != null) { perms.push(PermissionIndex.service_pause) } if (this.data?.bPublished) { // publish is an irreversible one-time operation perms.push(PermissionIndex.service_publish) } if (this.data?.endpoint != null) { perms.push(PermissionIndex.service_endpoint) } if (this.data?.repository != null) { perms.push(PermissionIndex.service_repository) } if (this.data?.clone_new != null) { perms.push(PermissionIndex.service_clone) } if (this.data?.gen_discount != null) { perms.push(PermissionIndex.service_discount_transfer) } if (this.data?.arbitration != null) { perms.push(PermissionIndex.service_arbitration) } if (this.data?.buy_guard != null) { perms.push(PermissionIndex.service_buyer_guard) } if (this.data?.endpoint != null) { perms.push(PermissionIndex.service_endpoint) } if (this.data?.extern_withdraw_treasury != null) { perms.push(PermissionIndex.service_treasury) } if (this.data?.machine != null) { perms.push(PermissionIndex.service_machine) } if (this.data?.payee_treasury != null && this.object_address) { perms.push(PermissionIndex.service_payee) } if (this.data?.withdraw_guard != null) { perms.push(PermissionIndex.service_withdraw_guards) } if (this.data?.refund_guard != null) { perms.push(PermissionIndex.service_refund_guards) } if (this.data?.customer_required_info != null) { perms.push(PermissionIndex.service_customer_required) } if (this.data?.sales != null) { perms.push(PermissionIndex.service_sales) } if (this.data?.order_new != null) { if (this.object_address) { if ((this.content as ObjectService)?.buy_guard) { guards.push((this.content as ObjectService).buy_guard!) } } } if ((this.data?.order_refund as RefundWithGuard)?.refund_guard != null) { const guard = await LocalMark.Instance().get_address((this.data?.order_refund as RefundWithGuard)?.refund_guard); if (guard) guards.push(guard); } if (this.data.order_withdrawl != null) { // permission(may be guard) + withdraw_guard perms.push(PermissionIndex.service_withdraw) } if (typeof(this.data?.order_withdrawl?.data?.withdraw_guard) === 'string') { const guard = await LocalMark.Instance().get_address(this.data?.order_withdrawl?.data?.withdraw_guard); if (guard) guards.push(guard); } return await this.check_permission_and_call(this.permission_address, perms, guards, checkOwner, undefined, account) } return await this.exec(account); } private order_allowed() : boolean { if ((this.content as ObjectService)?.bPaused) { ERROR(Errors.InvalidParam, 'Service is paused'); return false; } if (((this.content as ObjectService)?.bPublished !== true)) { ERROR(Errors.InvalidParam, 'Service is not published'); return false; } return true } private order_progress = async (order?: string, order_new?:OrderResult) : Promise<ProgressObject> => { if (order) { const r = await query_objects({objects:[order]}); if (r?.objects?.length !== 1 || r?.objects[0]?.type !== 'Order') { ERROR(Errors.InvalidParam, 'order_progress:' + order); } return (r.objects[0] as ObjectOrder).progress! as ProgressObject; } else if (order_new) { return order_new.progress as ProgressObject; } else { ERROR(Errors.InvalidParam, 'order_progress'); } } protected async operate (txb:TransactionBlock, passport?:PassportObject, account?:string) { let obj : Service | undefined ; let perm: Permission | undefined; let permission : PermissionObject | undefined; let payee: Treasury | undefined; if (this.object_address) { obj = Service.From(txb, this.type_parameter!, this.permission_address!, this.object_address); permission = this.permission_address; } else { const n = GetObjectMain(this.data.object) as TypeNamedObjectWithPermission; permission = await LocalMark.Instance().get_address(GetObjectExisted(n?.permission)); if (!permission) { perm = Permission.New(txb, GetObjectParam(n?.permission)?.description ?? ''); permission = perm.get_object(); } const treasury_address = await LocalMark.Instance().get_address(GetObjectExisted(this.data.payee_treasury)); if (!treasury_address) { payee = Treasury.New(txb, this.type_parameter!, permission, GetObjectParam(this.data.payee_treasury)?.description ?? '', perm?undefined:passport); } const t = payee ? payee.get_object() : treasury_address; if (!t) { ERROR(Errors.InvalidParam, 'CallService_Data.payee_treasury:' + (this.data.payee_treasury as any).address); } obj = Service.New(txb, this.type_parameter!, permission, this.data?.description??'', t, perm?undefined:passport); } if (!obj) ERROR(Errors.InvalidParam, 'CallService_Data.object:' + this.object_address); if (!permission) ERROR(Errors.InvalidParam, 'CallService_Data.permission:' + this.permission_address); const pst = perm?undefined:passport; var order_new : OrderResult | undefined; if (this.data?.order_new != null && this.order_allowed()) { let b = BigInt(0); let coin : any; this.data.order_new.buy_items.forEach(v => { b += BigInt(v.max_price) * BigInt(v.count) }); coin = await Account.Instance().get_coin_object(txb, b, account, this.type_parameter); if (coin) { order_new = obj.order(this.data.order_new.buy_items, coin, await LocalMark.Instance().get_address(this.data.order_new.discount_object), (this?.content as ObjectService).machine!, await this.info_crypto(this.data.order_new.customer_info_required), pst); } } if (this.data?.order_agent != null) { const o = this.data.order_agent.order ? await LocalMark.Instance().get_address(this.data.order_agent.order) : order_new?.order; if (!o) ERROR(Errors.InvalidParam, `CallService_Data.data.order_agent.order:${this.data.order_agent.order}`); const p = await this.order_progress(this.data.order_agent.order, order_new); const agents = await GetManyAccountOrMark_Address(this.data.order_agent.agents); obj?.set_order_agent(o, agents.filter((v):v is string =>v!==undefined), p); } if (this.data?.order_required_info?.customer_info_required) { const o = await LocalMark.Instance().get_address(this.data.order_required_info.order); if (!o) ERROR(Errors.InvalidParam, `CallService_Data.data.order_required_info.order:${this.data.order_required_info.order}`); const crypto = await this.info_crypto(this.object_address, this.data.order_required_info.customer_info_required); if (crypto) { obj?.update_order_required_info(o, crypto); } } if (this.data?.order_refund != null) { const o = await LocalMark.Instance().get_address(this.data.order_refund.order); if (!o) ERROR(Errors.InvalidParam, `CallService_Data.data.order_agent.order:${this.data.order_refund.order}`); if ((this.data?.order_refund as any)?.arb) { const r = await query_objects({objects:[(this.data?.order_refund as any)?.arb]}); if (r?.objects?.length!== 1 || r?.objects[0]?.type!== 'Arb') { ERROR(Errors.InvalidParam, 'order_refund.arb:' + (this.data?.order_refund as any)?.arb); } obj?.refund_withArb(o!, r?.objects[0].object, Arbitration.parseArbObjectType(r.objects[0].type_raw)!); } else { const guard = await LocalMark.Instance().get_address((this.data?.order_refund as RefundWithGuard)?.refund_guard); if (guard) obj?.refund(o!, guard, pst) } } if (this.data?.order_withdrawl != null && pst) { //@ need withdrawal pst const n = this.data?.order_withdrawl; const o = await LocalMark.Instance().get_address(n.order); if (!o) ERROR(Errors.InvalidParam, `CallService_Data.data.order_agent.order:${this.data.order_withdrawl.order}`); const [for_guard, for_object, withdrawGuard] = await LocalMark.Instance().get_many_address( [n.data.for_guard, n.data.for_object, n.data.withdraw_guard]); if (!withdrawGuard) ERROR(Errors.InvalidParam, `CallService_Data.data.order_withdrawl.data.withdraw_guard:${this.data.order_withdrawl.data.withdraw_guard}`); obj?.withdraw(o!, {withdraw_guard:withdrawGuard, treasury:(this.content as ObjectService).payee_treasury!, index: n.data.index, for_guard:for_guard, for_object:for_object, remark:n.data.remark}, pst); } if (this.data?.order_payer != null) { const o = this.data.order_payer.order ? await LocalMark.Instance().get_address(this.data.order_payer.order) : order_new?.order; if (!o) ERROR(Errors.InvalidParam, `CallService_Data.data.order_agent.order:${this.data.order_payer.order}`); const p = await this.order_progress(this.data.order_payer.order, order_new); const payer = await GetAccountOrMark_Address(this.data.order_payer.payer_new); if (payer) obj?.change_order_payer(o, payer, p) } if (order_new && this?.data?.order_new) { const buy = obj.order_launch(order_new); await this.new_with_mark('Order', txb, buy.order, (this.data?.order_new as any)?.namedNewOrder, account, [TagName.Launch, TagName.Order]); if (buy?.progress) { await this.new_with_mark('Progress', txb, buy.progress, (this.data?.order_new as any)?.namedNewProgress, account, [TagName.Launch, 'progress']); } } if (this.data?.description != null && this.object_address) { obj?.set_description(this.data.description, pst); } if (this.data?.endpoint != null) { obj?.set_endpoint(this.data.endpoint, pst) } if (this.data?.payee_treasury != null && this.object_address) { const treasury_address = await LocalMark.Instance().get_address(GetObjectExisted(this.data.payee_treasury)); if (!treasury_address) { payee = Treasury.New(txb, this.type_parameter!, permission, GetObjectParam(this.data.payee_treasury)?.description ?? '', permission?undefined:passport); } const t = payee ? payee.get_object() : treasury_address; if (!t) { ERROR(Errors.InvalidParam, 'CallService_Data.payee_treasury:' + (this.data.payee_treasury as any).address); } obj?.set_payee(t, pst); } if (this.data?.gen_discount != null) { const add: WowokDiscountDispatch[] = []; for (let i = 0; i < this.data.gen_discount.length; ++ i) { let v = this.data.gen_discount[i]; const addr = await GetAccountOrMark_Address(v.receiver); if (addr) { add.push({receiver:addr, count:v.count ?? 1, discount:v.discount}) } } obj?.discount_transfer(add, pst) } if (this.data?.repository != null) { switch (this.data.repository.op) { case 'add': case 'set': if (this.data.repository.op === 'set') obj?.remove_repository([], true, pst); for (let i = 0; i < this.data.repository.objects.length; ++ i) { let v = this.data.repository.objects[i]; const addr = await LocalMark.Instance().get_address(v); if (addr) { obj?.add_repository(addr, pst) } } break; case 'remove': obj?.remove_repository(await LocalMark.Instance().get_many_address2(this.data.repository.objects), false, pst) break; case 'removeall': obj?.remove_repository([], true, pst) break; } } if (this.data?.extern_withdraw_treasury != null) { switch(this.data.extern_withdraw_treasury.op) { case 'add': case 'set': if (this.data.extern_withdraw_treasury.op === 'set') obj?.remove_treasury([], true, pst); const r = await query_objects({objects:this.data.extern_withdraw_treasury.objects, no_cache:true}); r.objects?.forEach(v => { if (v.type ==='Treasury') { obj?.add_treasury(v.object, Treasury.parseObjectType(v.type_raw), pst) } }); break; case 'remove': obj?.remove_treasury(await LocalMark.Instance().get_many_address2(this.data.extern_withdraw_treasury.objects), false, pst) break; case 'removeall': obj?.remove_treasury([], false, pst) break; } } if (this.data?.machine != null) { const machine = await LocalMark.Instance().get_address(this.data.machine); obj?.set_machine(machine, pst) } if (this.data?.arbitration != null) { switch(this.data.arbitration.op) { case 'add': case 'set': if (this.data.arbitration.op === 'set') obj?.remove_arbitration([], true, pst); const r = await query_objects({objects:this.data.arbitration.objects, no_cache:true}); r.objects?.forEach(v => { if (v.type ==='Arbitration') { obj?.add_arbitration(v.object, Arbitration.parseObjectType(v.type_raw), pst) } }); break; case 'remove': obj?.remove_arbitration(await LocalMark.Instance().get_many_address2(this.data.arbitration.objects), false, pst) break; case 'removeall': obj?.remove_arbitration([], false, pst) break; } } if (this.data?.customer_required_info != null) { if (this.data.customer_required_info.required_info.length > 0 && this.data.customer_required_info.pubkey) { obj?.set_customer_required(this.data.customer_required_info.pubkey, this.data.customer_required_info.required_info, pst); } else if (this.data.customer_required_info.pubkey) { obj?.change_required_pubkey(this.data.customer_required_info.pubkey, pst); } } if (this.data?.sales != null) { switch(this.data.sales.op) { case 'add': obj?.add_sales(this.data.sales.sales, false, pst) break; case 'remove': obj?.remove_sales(this.data.sales.sales_name, pst) break; } } if (this.data?.withdraw_guard != null) { switch(this.data.withdraw_guard.op) { case 'add': case 'set': if (this.data.withdraw_guard.op === 'set') obj?.remove_withdraw_guards([], true, pst); const add = [] for (let i = 0; i < this.data.withdraw_guard.guards.length; ++ i) { let v = this.data.withdraw_guard.guards[i]; const addr = typeof(v.guard) === 'string' ? await LocalMark.Instance().get_address(v.guard as string) : v.guard; if (addr) { v.guard = addr; add.push(v) } } obj?.add_withdraw_guards(add, pst) break; case 'remove': obj?.remove_withdraw_guards(await LocalMark.Instance().get_many_address2(this.data.withdraw_guard.guards), false, pst) break; case 'removeall': obj?.remove_withdraw_guards([], true, pst) break; } } if (this.data?.refund_guard != null) { switch(this.data.refund_guard.op) { case 'add': case 'set': if (this.data.refund_guard.op === 'set') obj?.remove_refund_guards([], true, pst); const add = [] for (let i = 0; i < this.data.refund_guard.guards.length; ++ i) { let v = this.data.refund_guard.guards[i]; const addr = typeof(v.guard) === 'string' ? await LocalMark.Instance().get_address(v.guard as string) : v.guard; if (addr) { v.guard = addr; add.push(v) } } obj?.add_refund_guards(add, pst) break; case 'remove': obj?.remove_refund_guards(await LocalMark.Instance().get_many_address2(this.data.refund_guard.guards), false, pst) break; case 'removeall': obj?.remove_refund_guards([], true, pst) break; } } if (this.data?.bPublished) { obj?.publish(pst) } if (this.data?.buy_guard != null) { obj?.set_buy_guard(this.data.buy_guard, pst) } if (this.data?.bPaused != null) { obj?.pause(this.data.bPaused, pst) } if (this.data?.clone_new != null) { await this.new_with_mark('Service', txb, obj.clone(this.data.clone_new?.token_type_new, true, pst) as TxbAddress, (this.data?.clone_new as any)?.namedNew, account); } if (payee) { await this.new_with_mark('Treasury', txb, payee.launch(), GetObjectParam(this.data?.payee_treasury), account); } if (perm) { const n = GetObjectMain(this.data.object) as TypeNamedObjectWithPermission; await this.new_with_mark('Permission', txb, perm.launch(), GetObjectParam(n?.permission), account); } if (!this.object_address) { await this.new_with_mark('Service', txb, obj.launch(), GetObjectMain(this.data?.object), account); } } private info_crypto = async (object?:string, info?: string) : Promise<Customer_RequiredInfo | undefined>=> { if (!this.content && info && object) { await this.update_content('Service', object); } const pubkey = (this.content as ObjectService).customer_required_info?.pubkey ?? ''; var info_crypto: Customer_RequiredInfo | undefined ; if (pubkey && info) { info_crypto = { customer_pubkey: pubkey, customer_info_crypt: crypto_string(info, pubkey) } } return info_crypto } }