tx · DQM4mvP7FFZoA2cn95LkxyXeVAB24StMsnEUx41mLtgK

3N9be2mwrA52WJho6DiesZkk4351GvpnWuj:  -0.04000000 Waves

2022.07.29 14:05 [2161025] smart account 3N9be2mwrA52WJho6DiesZkk4351GvpnWuj > SELF 0.00000000 Waves

{ "type": 13, "id": "DQM4mvP7FFZoA2cn95LkxyXeVAB24StMsnEUx41mLtgK", "fee": 4000000, "feeAssetId": null, "timestamp": 1659092775314, "version": 1, "sender": "3N9be2mwrA52WJho6DiesZkk4351GvpnWuj", "senderPublicKey": "6mzmbCza9iqbzxMEELcEA4Xc9NeF4CYpbTtz1zMK3C7x", "proofs": [ "3e74RuqDgs7kQmXKE9Ti7CJEZbFGJJL7XZMfKFRHPRfyym1Gto4de5C4DSRe1S6a8CZwCy4oc9ZXES6xyHNAiYMK", "mfUgXQRvhHovy8UGTtMvqSfnPaNF9w6422ygFFfq5V7zBMo9yCmqdGhbKaQ9oDMgtX9PcvZp4wGpF9mjoLqTJ6M", "3RHi7WGjScWWP5jNHfpuM4HF6yjzMBLGcumobeb119itde3bTMHVcBXd9f39NbMqTJmJfYcecCJsQ77pfchWCJSw" ], "script": "base64:BgJBCAISDgoMCAgICAgIAQEBAQEBEgUKAwgIARIAEgASBQoDCAEIEgMKAQESBAoCAQgSABIECgIIARIECgIIARIAEgCIAQEPZ2V0U3RyaW5nT3JGYWlsAgdhZGRyZXNzA2tleQkBE3ZhbHVlT3JFcnJvck1lc3NhZ2UCCQCdCAIFB2FkZHJlc3MFA2tleQkAuQkCCQDMCAICCm1hbmRhdG9yeSAJAMwIAgkApQgBBQdhZGRyZXNzCQDMCAICAS4JAMwIAgUDa2V5CQDMCAICDyBpcyBub3QgZGVmaW5lZAUDbmlsAgABBWxjYWxjAQFsCQC5CAEFAWwBDmdldE51bWJlckJ5S2V5AQNrZXkJAQt2YWx1ZU9yRWxzZQIJAJoIAgUEdGhpcwUDa2V5AAABDmdldFN0cmluZ0J5S2V5AQNrZXkJAQt2YWx1ZU9yRWxzZQIJAJ0IAgUEdGhpcwUDa2V5AgABDGdldEJvb2xCeUtleQEDa2V5CQELdmFsdWVPckVsc2UCCQCbCAIFBHRoaXMFA2tleQcBGGdldE51bWJlckJ5QWRkcmVzc0FuZEtleQIHYWRkcmVzcwNrZXkJAQt2YWx1ZU9yRWxzZQIJAJoIAgUHYWRkcmVzcwUDa2V5AAABGGdldFN0cmluZ0J5QWRkcmVzc0FuZEtleQIHYWRkcmVzcwNrZXkJAQt2YWx1ZU9yRWxzZQIJAJ0IAgkBEUBleHRyTmF0aXZlKDEwNjIpAQUHYWRkcmVzcwUDa2V5AgABFmdldEJvb2xCeUFkZHJlc3NBbmRLZXkCB2FkZHJlc3MDa2V5CQELdmFsdWVPckVsc2UCCQCbCAIFB2FkZHJlc3MFA2tleQcBCWFzQW55TGlzdAEBdgQHJG1hdGNoMAUBdgMJAAECBQckbWF0Y2gwAglMaXN0W0FueV0EAWwFByRtYXRjaDAFAWwJAAIBAhtmYWlsIHRvIGNhc3QgaW50byBMaXN0W0FueV0BCGFzU3RyaW5nAQF2BAckbWF0Y2gwBQF2AwkAAQIFByRtYXRjaDACBlN0cmluZwQBcwUHJG1hdGNoMAUBcwkAAgECGGZhaWwgdG8gY2FzdCBpbnRvIFN0cmluZwEFYXNJbnQBAXYEByRtYXRjaDAFAXYDCQABAgUHJG1hdGNoMAIDSW50BAFpBQckbWF0Y2gwBQFpCQACAQIVZmFpbCB0byBjYXN0IGludG8gSW50AQlhc1BheW1lbnQBAXYEByRtYXRjaDAFAXYDCQABAgUHJG1hdGNoMAIPQXR0YWNoZWRQYXltZW50BAFwBQckbWF0Y2gwBQFwCQACAQIhZmFpbCB0byBjYXN0IGludG8gQXR0YWNoZWRQYXltZW50ARJhc1N3YXBQYXJhbXNTVFJVQ1QBAXYEByRtYXRjaDAFAXYDCQABAgUHJG1hdGNoMAIZKEludCwgSW50LCBJbnQsIEludCwgSW50KQQGc3RydWN0BQckbWF0Y2gwBQZzdHJ1Y3QJAAIBAhVmYWlsIHRvIGNhc3QgaW50byBJbnQAA1NFUAICX18AB1dBVkVMRVQAgMLXLwAFUEFVTEkAwIQ9AAhQUklDRUxFVADAhD0ADkRFRkFVTFRTV0FQRkVFAKCcAQAMSWR4TmV0QW1vdW50AAAADElkeEZlZUFtb3VudAABAA5JZHhHcm9zc0Ftb3VudAACABlJZHhDb250cm9sQ2ZnTmV1dHJpbm9EYXBwAAEAGElkeENvbnRyb2xDZmdBdWN0aW9uRGFwcAACABRJZHhDb250cm9sQ2ZnUnBkRGFwcAADABVJZHhDb250cm9sQ2ZnTWF0aERhcHAABAAcSWR4Q29udHJvbENmZ0xpcXVpZGF0aW9uRGFwcAAFABVJZHhDb250cm9sQ2ZnUmVzdERhcHAABgAdSWR4Q29udHJvbENmZ05vZGVSZWdpc3RyeURhcHAABwAcSWR4Q29udHJvbENmZ05zYnRTdGFraW5nRGFwcAAIABlJZHhDb250cm9sQ2ZnTWVkaWF0b3JEYXBwAAkAHElkeENvbnRyb2xDZmdTdXJmU3Rha2luZ0RhcHAACgAgSWR4Q29udHJvbENmZ0duc2J0Q29udHJvbGxlckRhcHAACwERa2V5Q29udHJvbEFkZHJlc3MAAhwlcyVzX19jb25maWdfX2NvbnRyb2xBZGRyZXNzAQ1rZXlDb250cm9sQ2ZnAAIRJXNfX2NvbnRyb2xDb25maWcBFHJlYWRDb250cm9sQ2ZnT3JGYWlsAQdjb250cm9sCQC1CQIJAQ9nZXRTdHJpbmdPckZhaWwCBQdjb250cm9sCQENa2V5Q29udHJvbENmZwAFA1NFUAEYZ2V0Q29udHJhY3RBZGRyZXNzT3JGYWlsAgpjb250cm9sQ2ZnA2lkeAkBE3ZhbHVlT3JFcnJvck1lc3NhZ2UCCQCmCAEJAJEDAgUKY29udHJvbENmZwUDaWR4CQCsAgICLUNvbnRyb2wgY2ZnIGRvZXNuJ3QgY29udGFpbiBhZGRyZXNzIGF0IGluZGV4IAkApAMBBQNpZHgAD2NvbnRyb2xDb250cmFjdAkBEUBleHRyTmF0aXZlKDEwNjIpAQkBC3ZhbHVlT3JFbHNlAgkAnQgCBQR0aGlzCQERa2V5Q29udHJvbEFkZHJlc3MAAiMzTjROUzdkNEpvOWE2RjE0TGlGVUtLWVZkVWtrZjJlUDRaeAAKY29udHJvbENmZwkBFHJlYWRDb250cm9sQ2ZnT3JGYWlsAQUPY29udHJvbENvbnRyYWN0AAxtYXRoQ29udHJhY3QJARhnZXRDb250cmFjdEFkZHJlc3NPckZhaWwCBQpjb250cm9sQ2ZnBRVJZHhDb250cm9sQ2ZnTWF0aERhcHAAE25zYnRTdGFraW5nQ29udHJhY3QJARhnZXRDb250cmFjdEFkZHJlc3NPckZhaWwCBQpjb250cm9sQ2ZnBRxJZHhDb250cm9sQ2ZnTnNidFN0YWtpbmdEYXBwABNzdXJmU3Rha2luZ0NvbnRyYWN0CQEYZ2V0Q29udHJhY3RBZGRyZXNzT3JGYWlsAgUKY29udHJvbENmZwUcSWR4Q29udHJvbENmZ1N1cmZTdGFraW5nRGFwcAAXZ25zYnRDb250cm9sbGVyQ29udHJhY3QJARhnZXRDb250cmFjdEFkZHJlc3NPckZhaWwCBQpjb250cm9sQ2ZnBSBJZHhDb250cm9sQ2ZnR25zYnRDb250cm9sbGVyRGFwcAAPYXVjdGlvbkNvbnRyYWN0CQEYZ2V0Q29udHJhY3RBZGRyZXNzT3JGYWlsAgUKY29udHJvbENmZwUYSWR4Q29udHJvbENmZ0F1Y3Rpb25EYXBwABJOZXV0cmlub0Fzc2V0SWRLZXkCEW5ldXRyaW5vX2Fzc2V0X2lkAA5Cb25kQXNzZXRJZEtleQINYm9uZF9hc3NldF9pZAASQXVjdGlvbkNvbnRyYWN0S2V5AhBhdWN0aW9uX2NvbnRyYWN0ABZOc2J0U3Rha2luZ0NvbnRyYWN0S2V5AhNuc2J0U3Rha2luZ0NvbnRyYWN0ABZMaXF1aWRhdGlvbkNvbnRyYWN0S2V5AhRsaXF1aWRhdGlvbl9jb250cmFjdAAOUlBEQ29udHJhY3RLZXkCDHJwZF9jb250cmFjdAARQ29udG9sQ29udHJhY3RLZXkCEGNvbnRyb2xfY29udHJhY3QAD01hdGhDb250cmFjdEtleQINbWF0aF9jb250cmFjdAAbQmFsYW5jZVdhdmVzTG9ja0ludGVydmFsS2V5AhtiYWxhbmNlX3dhdmVzX2xvY2tfaW50ZXJ2YWwAHkJhbGFuY2VOZXV0cmlub0xvY2tJbnRlcnZhbEtleQIeYmFsYW5jZV9uZXV0cmlub19sb2NrX2ludGVydmFsABVNaW5XYXZlc1N3YXBBbW91bnRLZXkCFW1pbl93YXZlc19zd2FwX2Ftb3VudAAYTWluTmV1dHJpbm9Td2FwQW1vdW50S2V5AhhtaW5fbmV1dHJpbm9fc3dhcF9hbW91bnQAG05vZGVPcmFjbGVQcm92aWRlclB1YktleUtleQIUbm9kZV9vcmFjbGVfcHJvdmlkZXIAFU5ldXRyaW5vT3V0RmVlUGFydEtleQIYbmV1dHJpbm9PdXRfc3dhcF9mZWVQYXJ0ABJXYXZlc091dEZlZVBhcnRLZXkCFXdhdmVzT3V0X3N3YXBfZmVlUGFydAEPa2V5Tm9kZVJlZ2lzdHJ5AQdhZGRyZXNzCQCsAgICBCVzX18FB2FkZHJlc3MACFByaWNlS2V5AgVwcmljZQANUHJpY2VJbmRleEtleQILcHJpY2VfaW5kZXgADElzQmxvY2tlZEtleQIKaXNfYmxvY2tlZAESZ2V0UHJpY2VIaXN0b3J5S2V5AQVibG9jawkArAICCQCsAgIFCFByaWNlS2V5AgFfCQCkAwEFBWJsb2NrARhnZXRIZWlnaHRQcmljZUJ5SW5kZXhLZXkBBWluZGV4CQCsAgIJAKwCAgUNUHJpY2VJbmRleEtleQIBXwkApAMBBQVpbmRleAEVZ2V0U3Rha2luZ05vZGVCeUluZGV4AQNpZHgJAQ5nZXRTdHJpbmdCeUtleQEJALkJAgkAzAgCAgYlcyVkJXMJAMwIAgIFbGVhc2UJAMwIAgkApAMBBQNpZHgJAMwIAgILbm9kZUFkZHJlc3MFA25pbAUDU0VQARxnZXRTdGFraW5nTm9kZUFkZHJlc3NCeUluZGV4AQNpZHgJARFAZXh0ck5hdGl2ZSgxMDYyKQEJARVnZXRTdGFraW5nTm9kZUJ5SW5kZXgBBQNpZHgBH2dldFJlc2VydmVkQW1vdW50Rm9yU3BvbnNvcnNoaXAACQELdmFsdWVPckVsc2UCCQCaCAIFBHRoaXMJALkJAgkAzAgCAgQlcyVzCQDMCAICBWxlYXNlCQDMCAICF3Nwb25zb3JzaGlwV2F2ZXNSZXNlcnZlBQNuaWwFA1NFUAkAaAIA6AcFB1dBVkVMRVQBGGdldEJhbGFuY2VVbmxvY2tCbG9ja0tleQEFb3duZXIJAKwCAgIVYmFsYW5jZV91bmxvY2tfYmxvY2tfBQVvd25lcgENZ2V0TGVhc2VJZEtleQEJbm9kZUluZGV4CQC5CQIJAMwIAgIGJXMlZCVzCQDMCAICBWxlYXNlCQDMCAIJAKQDAQUJbm9kZUluZGV4CQDMCAICAmlkBQNuaWwFA1NFUAEWZ2V0TGVhc2VJZEJ5QWRkcmVzc0tleQELbm9kZUFkZHJlc3MJALkJAgkAzAgCAgYlcyVzJXMJAMwIAgIObGVhc2VCeUFkZHJlc3MJAMwIAgULbm9kZUFkZHJlc3MJAMwIAgICaWQFA25pbAUDU0VQARFnZXRMZWFzZUFtb3VudEtleQEJbm9kZUluZGV4CQC5CQIJAMwIAgIGJXMlZCVzCQDMCAICBWxlYXNlCQDMCAIJAKQDAQUJbm9kZUluZGV4CQDMCAICBmFtb3VudAUDbmlsBQNTRVABGmdldExlYXNlQW1vdW50QnlBZGRyZXNzS2V5AQtub2RlQWRkcmVzcwkAuQkCCQDMCAICBiVzJXMlcwkAzAgCAg5sZWFzZUJ5QWRkcmVzcwkAzAgCBQtub2RlQWRkcmVzcwkAzAgCAgZhbW91bnQFA25pbAUDU0VQARhnZXRMZWFzZUdyb3VwTm9kZUxpc3RLZXkBCGdyb3VwTnVtCQC5CQIJAMwIAgIGJXMlZCVzCQDMCAICCmxlYXNlR3JvdXAJAMwIAgkApAMBBQhncm91cE51bQkAzAgCAghub2RlTGlzdAUDbmlsBQNTRVABEG1pblN3YXBBbW91bnRLRVkBCHN3YXBUeXBlCQCsAgIJAKwCAgIEbWluXwUIc3dhcFR5cGUCDF9zd2FwX2Ftb3VudAEOdG90YWxMb2NrZWRLRVkBCHN3YXBUeXBlCQCsAgICDWJhbGFuY2VfbG9ja18FCHN3YXBUeXBlARR0b3RhbExvY2tlZEJ5VXNlcktFWQIIc3dhcFR5cGUFb3duZXIJALkJAgkAzAgCAgxiYWxhbmNlX2xvY2sJAMwIAgUIc3dhcFR5cGUJAMwIAgUFb3duZXIFA25pbAIBXwEWYmFsYW5jZUxvY2tJbnRlcnZhbEtFWQEIc3dhcFR5cGUJAKwCAgkArAICAghiYWxhbmNlXwUIc3dhcFR5cGUCDl9sb2NrX2ludGVydmFsARpub2RlQmFsYW5jZUxvY2tJbnRlcnZhbEtFWQACGmJhbGFuY2Vfbm9kZV9sb2NrX2ludGVydmFsAQ1vdXRGZWVQYXJ0S0VZAQhzd2FwVHlwZQkArAICBQhzd2FwVHlwZQIQT3V0X3N3YXBfZmVlUGFydAERc3dhcHNUaW1lZnJhbWVLRVkAAg9zd2Fwc190aW1lZnJhbWUBEW1pblN3YXBBbW91bnRSRUFEAQhzd2FwVHlwZQkBC3ZhbHVlT3JFbHNlAgkAmggCBQR0aGlzCQEQbWluU3dhcEFtb3VudEtFWQEFCHN3YXBUeXBlAAABEnN3YXBzVGltZWZyYW1lUkVBRAAJAQt2YWx1ZU9yRWxzZQIJAJoIAgUEdGhpcwkBEXN3YXBzVGltZWZyYW1lS0VZAACgCwEPdG90YWxMb2NrZWRSRUFEAQhzd2FwVHlwZQkBC3ZhbHVlT3JFbHNlAgkAmggCBQR0aGlzCQEOdG90YWxMb2NrZWRLRVkBBQhzd2FwVHlwZQAAARV0b3RhbExvY2tlZEJ5VXNlclJFQUQCCHN3YXBUeXBlBW93bmVyCQELdmFsdWVPckVsc2UCCQCaCAIFBHRoaXMJARR0b3RhbExvY2tlZEJ5VXNlcktFWQIFCHN3YXBUeXBlBQVvd25lcgAAARdiYWxhbmNlTG9ja0ludGVydmFsUkVBRAEIc3dhcFR5cGUJAQt2YWx1ZU9yRWxzZQIJAJoIAgUEdGhpcwkBFmJhbGFuY2VMb2NrSW50ZXJ2YWxLRVkBBQhzd2FwVHlwZQCgCwEbbm9kZUJhbGFuY2VMb2NrSW50ZXJ2YWxSRUFEAAkBC3ZhbHVlT3JFbHNlAgkAmggCBQR0aGlzCQEabm9kZUJhbGFuY2VMb2NrSW50ZXJ2YWxLRVkAAAEBGGtleVN3YXBVc2VyU3BlbnRJblBlcmlvZAELdXNlckFkZHJlc3MJALkJAgkAzAgCAgQlcyVzCQDMCAICFXN3YXBVc2VyU3BlbnRJblBlcmlvZAkAzAgCBQt1c2VyQWRkcmVzcwUDbmlsBQNTRVABFWtleVVzZXJMYXN0U3dhcEhlaWdodAELdXNlckFkZHJlc3MJALkJAgkAzAgCAgQlcyVzCQDMCAICEnVzZXJMYXN0U3dhcEhlaWdodAkAzAgCBQt1c2VyQWRkcmVzcwUDbmlsBQNTRVABFmNvbnZlcnROZXV0cmlub1RvV2F2ZXMCBmFtb3VudAVwcmljZQkAawMJAGsDBQZhbW91bnQFCFBSSUNFTEVUBQVwcmljZQUHV0FWRUxFVAUFUEFVTEkBFmNvbnZlcnRXYXZlc1RvTmV1dHJpbm8CBmFtb3VudAVwcmljZQkAawMJAGsDBQZhbW91bnQFBXByaWNlBQhQUklDRUxFVAUFUEFVTEkFB1dBVkVMRVQBEmNvbnZlcnRXYXZlc1RvQm9uZAIGYW1vdW50BXByaWNlCQEWY29udmVydFdhdmVzVG9OZXV0cmlubwIFBmFtb3VudAUFcHJpY2UBFmNvbnZlcnRKc29uQXJyYXlUb0xpc3QBCWpzb25BcnJheQkAtQkCBQlqc29uQXJyYXkCASwBEW1pblN3YXBBbW91bnRGQUlMAghzd2FwVHlwZQ1taW5Td2FwQW1vdW50CQACAQkArAICCQCsAgIJAKwCAgIYVGhlIHNwZWNpZmllZCBhbW91bnQgaW4gBQhzd2FwVHlwZQIrIHN3YXAgaXMgbGVzcyB0aGFuIHRoZSByZXF1aXJlZCBtaW5pbXVtIG9mIAkApAMBBQ1taW5Td2FwQW1vdW50ARVlbWVyZ2VuY3lTaHV0ZG93bkZBSUwACQACAQJaY29udHJhY3QgaXMgYmxvY2tlZCBieSBFTUVSR0VOQ1kgU0hVVERPV04gYWN0aW9ucyB1bnRpbGwgcmVhY3RpdmF0aW9uIGJ5IGVtZXJnZW5jeSBvcmFjbGVzAQ5wcmljZUluZGV4RkFJTAUFaW5kZXgKcHJpY2VJbmRleAtpbmRleEhlaWdodAx1bmxvY2tIZWlnaHQPcHJldkluZGV4SGVpZ2h0CQACAQkArAICCQCsAgIJAKwCAgkArAICCQCsAgIJAKwCAgkArAICCQCsAgIJAKwCAgIjaW52YWxpZCBwcmljZSBoaXN0b3J5IGluZGV4OiBpbmRleD0JAKQDAQUFaW5kZXgCDCBwcmljZUluZGV4PQkApAMBBQpwcmljZUluZGV4Ag0gaW5kZXhIZWlnaHQ9CQCkAwEFC2luZGV4SGVpZ2h0Ag4gdW5sb2NrSGVpZ2h0PQkApAMBBQx1bmxvY2tIZWlnaHQCESBwcmV2SW5kZXhIZWlnaHQ9CQCkAwEFD3ByZXZJbmRleEhlaWdodAAPbmV1dHJpbm9Bc3NldElkCQDZBAEJAQ5nZXRTdHJpbmdCeUtleQEFEk5ldXRyaW5vQXNzZXRJZEtleQAKcHJpY2VJbmRleAkBGGdldE51bWJlckJ5QWRkcmVzc0FuZEtleQIFD2NvbnRyb2xDb250cmFjdAUNUHJpY2VJbmRleEtleQAJaXNCbG9ja2VkCQEWZ2V0Qm9vbEJ5QWRkcmVzc0FuZEtleQIFD2NvbnRyb2xDb250cmFjdAUMSXNCbG9ja2VkS2V5ABhub2RlT3JhY2xlUHJvdmlkZXJQdWJLZXkJANkEAQkBDmdldFN0cmluZ0J5S2V5AQUbTm9kZU9yYWNsZVByb3ZpZGVyUHViS2V5S2V5AAtib25kQXNzZXRJZAkA2QQBAixFeVpVdXlwUXAzV1doeHFMTng4WTZONkdzYlQyMWY1VUNyMzZYZDdnc0dDawAVZGVwcmVjYXRlZEJvbmRBc3NldElkCQDZBAECLDk3NWFrWkJmbk1qNTEzVTdNWmFIS3pRcm1zRXg1YUUzd2RXS1RySEJoYmpGABBuZXV0cmlub0NvbnRyYWN0BQR0aGlzAAxjdXJyZW50UHJpY2UJARhnZXROdW1iZXJCeUFkZHJlc3NBbmRLZXkCBQ9jb250cm9sQ29udHJhY3QFCFByaWNlS2V5ARtjaGVja0lzVmFsaWRNaW5TcG9uc29yZWRGZWUBAnR4BA5NSU5UUkFOU0ZFUkZFRQCgjQYEFlNwb25zb3JlZEZlZVVwcGVyQm91bmQA6AcED3JlYWxOZXV0cmlub0ZlZQkBFmNvbnZlcnRXYXZlc1RvTmV1dHJpbm8CBQ5NSU5UUkFOU0ZFUkZFRQUMY3VycmVudFByaWNlBA5taW5OZXV0cmlub0ZlZQkAaAIFD3JlYWxOZXV0cmlub0ZlZQACBA5tYXhOZXV0cmlub0ZlZQkAawMFD3JlYWxOZXV0cmlub0ZlZQUWU3BvbnNvcmVkRmVlVXBwZXJCb3VuZABkBAhpbnB1dEZlZQkBBXZhbHVlAQgFAnR4FG1pblNwb25zb3JlZEFzc2V0RmVlAwMJAGcCBQhpbnB1dEZlZQUObWluTmV1dHJpbm9GZWUJAGcCBQ5tYXhOZXV0cmlub0ZlZQUIaW5wdXRGZWUHCQAAAggFAnR4B2Fzc2V0SWQFD25ldXRyaW5vQXNzZXRJZAcBD2dldFByaWNlSGlzdG9yeQEFYmxvY2sJARhnZXROdW1iZXJCeUFkZHJlc3NBbmRLZXkCBQ9jb250cm9sQ29udHJhY3QJARJnZXRQcmljZUhpc3RvcnlLZXkBBQVibG9jawEVZ2V0SGVpZ2h0UHJpY2VCeUluZGV4AQVpbmRleAkBGGdldE51bWJlckJ5QWRkcmVzc0FuZEtleQIFD2NvbnRyb2xDb250cmFjdAkBGGdldEhlaWdodFByaWNlQnlJbmRleEtleQEFBWluZGV4ARZrZXlMb2NrUGFyYW1Vc2VyQW1vdW50AQt1c2VyQWRkcmVzcwkAuQkCCQDMCAICBiVzJXMlcwkAzAgCAgtwYXJhbUJ5VXNlcgkAzAgCBQt1c2VyQWRkcmVzcwkAzAgCAgZhbW91bnQFA25pbAUDU0VQAAxzSWR4U3dhcFR5cGUAAQAKc0lkeFN0YXR1cwACAAxzSWR4SW5BbW91bnQAAwAJc0lkeFByaWNlAAQAEHNJZHhPdXROZXRBbW91bnQABQAQc0lkeE91dEZlZUFtb3VudAAGAA9zSWR4U3RhcnRIZWlnaHQABwASc0lkeFN0YXJ0VGltZXN0YW1wAAgADXNJZHhFbmRIZWlnaHQACQAQc0lkeEVuZFRpbWVzdGFtcAAKABRzSWR4U2VsZlVubG9ja0hlaWdodAALABRzSWR4UmFuZFVubG9ja0hlaWdodAAMAAlzSWR4SW5kZXgADQAQc0lkeFdpdGhkcmF3VHhJZAAOAAtzSWR4TWluUmFuZAAPAAtzSWR4TWF4UmFuZAAQAQdzd2FwS0VZAgt1c2VyQWRkcmVzcwR0eElkCQC5CQIJAMwIAgIEJXMlcwkAzAgCBQt1c2VyQWRkcmVzcwkAzAgCBQR0eElkBQNuaWwFA1NFUAELc3RyU3dhcERBVEEQCHN3YXBUeXBlBnN0YXR1cwhpbkFtb3VudAVwcmljZQxvdXROZXRBbW91bnQMb3V0RmVlQW1vdW50C3N0YXJ0SGVpZ2h0DnN0YXJ0VGltZXN0YW1wCWVuZEhlaWdodAxlbmRUaW1lc3RhbXAQc2VsZlVubG9ja0hlaWdodBByYW5kVW5sb2NrSGVpZ2h0BWluZGV4DHdpdGhkcmF3VHhJZAdyYW5kTWluB3JhbmRNYXgJALkJAgkAzAgCAhwlcyVzJWQlZCVkJWQlZCVkJWQlZCVkJWQlZCVzCQDMCAIFCHN3YXBUeXBlCQDMCAIFBnN0YXR1cwkAzAgCBQhpbkFtb3VudAkAzAgCBQVwcmljZQkAzAgCBQxvdXROZXRBbW91bnQJAMwIAgUMb3V0RmVlQW1vdW50CQDMCAIFC3N0YXJ0SGVpZ2h0CQDMCAIFDnN0YXJ0VGltZXN0YW1wCQDMCAIFCWVuZEhlaWdodAkAzAgCBQxlbmRUaW1lc3RhbXAJAMwIAgUQc2VsZlVubG9ja0hlaWdodAkAzAgCBRByYW5kVW5sb2NrSGVpZ2h0CQDMCAIFBWluZGV4CQDMCAIFDHdpdGhkcmF3VHhJZAkAzAgCBQdyYW5kTWluCQDMCAIFB3JhbmRNYXgFA25pbAUDU0VQAQ9wZW5kaW5nU3dhcERBVEEDCHN3YXBUeXBlDWluQXNzZXRBbW91bnQQc2VsZlVubG9ja0hlaWdodAkBC3N0clN3YXBEQVRBEAUIc3dhcFR5cGUCB1BFTkRJTkcJAKQDAQUNaW5Bc3NldEFtb3VudAIBMAIBMAIBMAkApAMBBQZoZWlnaHQJAKQDAQgFCWxhc3RCbG9jawl0aW1lc3RhbXACATACATAJAKQDAQUQc2VsZlVubG9ja0hlaWdodAIBMAIBMAIETlVMTAIBMAIBMAEOZmluaXNoU3dhcERBVEEHCWRhdGFBcnJheQVwcmljZQxvdXROZXRBbW91bnQMb3V0RmVlQW1vdW50EHJhbmRVbmxvY2tIZWlnaHQFaW5kZXgMd2l0aGRyYXdUeElkCQELc3RyU3dhcERBVEEQCQCRAwIFCWRhdGFBcnJheQUMc0lkeFN3YXBUeXBlAghGSU5JU0hFRAkAkQMCBQlkYXRhQXJyYXkFDHNJZHhJbkFtb3VudAkApAMBBQVwcmljZQkApAMBBQxvdXROZXRBbW91bnQJAKQDAQUMb3V0RmVlQW1vdW50CQCRAwIFCWRhdGFBcnJheQUPc0lkeFN0YXJ0SGVpZ2h0CQCRAwIFCWRhdGFBcnJheQUSc0lkeFN0YXJ0VGltZXN0YW1wCQCkAwEFBmhlaWdodAkApAMBCAUJbGFzdEJsb2NrCXRpbWVzdGFtcAkAkQMCBQlkYXRhQXJyYXkFFHNJZHhTZWxmVW5sb2NrSGVpZ2h0CQCkAwEFEHJhbmRVbmxvY2tIZWlnaHQJAKQDAQUFaW5kZXgFDHdpdGhkcmF3VHhJZAkAkQMCBQlkYXRhQXJyYXkFC3NJZHhNaW5SYW5kCQCRAwIFCWRhdGFBcnJheQULc0lkeE1heFJhbmQBEnN3YXBEYXRhRmFpbE9yUkVBRAILdXNlckFkZHJlc3MIc3dhcFR4SWQEB3N3YXBLZXkJAQdzd2FwS0VZAgULdXNlckFkZHJlc3MFCHN3YXBUeElkCQC1CQIJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkAnQgCBQR0aGlzBQdzd2FwS2V5CQCsAgICEW5vIHN3YXAgZGF0YSBmb3IgBQdzd2FwS2V5BQNTRVABCWFwcGx5RmVlcwILYW1vdW50R3Jvc3MHZmVlUGFydAQJZmVlQW1vdW50CQBrAwULYW1vdW50R3Jvc3MFB2ZlZVBhcnQFBVBBVUxJCQDMCAIJAGUCBQthbW91bnRHcm9zcwUJZmVlQW1vdW50CQDMCAIFCWZlZUFtb3VudAkAzAgCBQthbW91bnRHcm9zcwUDbmlsAQNhYnMBAXgDCQBmAgAABQF4CQEBLQEFAXgFAXgBCnNlbGVjdE5vZGUBDXVubGVhc2VBbW91bnQEDWFtb3VudFRvTGVhc2UJAGUCCQBlAggJAO8HAQUQbmV1dHJpbm9Db250cmFjdAlhdmFpbGFibGUFDXVubGVhc2VBbW91bnQJAR9nZXRSZXNlcnZlZEFtb3VudEZvclNwb25zb3JzaGlwAAQKb2xkTGVhc2VkMAkBDmdldE51bWJlckJ5S2V5AQkBEWdldExlYXNlQW1vdW50S2V5AQAABApvbGRMZWFzZWQxCQEOZ2V0TnVtYmVyQnlLZXkBCQERZ2V0TGVhc2VBbW91bnRLZXkBAAEECm5ld0xlYXNlZDAJAGQCBQ1hbW91bnRUb0xlYXNlBQpvbGRMZWFzZWQwBApuZXdMZWFzZWQxCQBkAgUNYW1vdW50VG9MZWFzZQUKb2xkTGVhc2VkMQMDCQBmAgUKbmV3TGVhc2VkMAAABgkAZgIFCm5ld0xlYXNlZDEAAAQGZGVsdGEwCQEDYWJzAQkAZQIFCm5ld0xlYXNlZDAFCm9sZExlYXNlZDEEBmRlbHRhMQkBA2FicwEJAGUCBQpuZXdMZWFzZWQxBQpvbGRMZWFzZWQwAwkAZwIFBmRlbHRhMQUGZGVsdGEwCQCUCgIAAAUKbmV3TGVhc2VkMAkAlAoCAAEFCm5ld0xlYXNlZDEJAJQKAgD///////////8BAAABCHRoaXNPbmx5AQFpAwkBAiE9AggFAWkGY2FsbGVyBQR0aGlzCQACAQItUGVybWlzc2lvbiBkZW5pZWQ6IHRoaXMgY29udHJhY3Qgb25seSBhbGxvd2VkBgEWcHJlcGFyZVVubGVhc2VBbmRMZWFzZQENdW5sZWFzZUFtb3VudAQJbm9kZVR1cGxlCQEKc2VsZWN0Tm9kZQEFDXVubGVhc2VBbW91bnQECW5vZGVJbmRleAgFCW5vZGVUdXBsZQJfMQQObmV3TGVhc2VBbW91bnQIBQlub2RlVHVwbGUCXzIDCQBmAgUObmV3TGVhc2VBbW91bnQAAAQKbGVhc2VJZEtleQkBDWdldExlYXNlSWRLZXkBBQlub2RlSW5kZXgECG9sZExlYXNlCQCcCAIFBHRoaXMFCmxlYXNlSWRLZXkEDnVubGVhc2VPckVtcHR5AwkBCWlzRGVmaW5lZAEFCG9sZExlYXNlCQDMCAIJAQtMZWFzZUNhbmNlbAEJAQV2YWx1ZQEFCG9sZExlYXNlBQNuaWwFA25pbAQObGVhc2VBbW91bnRLZXkJARFnZXRMZWFzZUFtb3VudEtleQEFCW5vZGVJbmRleAQFbGVhc2UJAMQIAgkBHGdldFN0YWtpbmdOb2RlQWRkcmVzc0J5SW5kZXgBBQlub2RlSW5kZXgFDm5ld0xlYXNlQW1vdW50CQDOCAIFDnVubGVhc2VPckVtcHR5CQDMCAIFBWxlYXNlCQDMCAIJAQtCaW5hcnlFbnRyeQIFCmxlYXNlSWRLZXkJAQVsY2FsYwEFBWxlYXNlCQDMCAIJAQxJbnRlZ2VyRW50cnkCCQERZ2V0TGVhc2VBbW91bnRLZXkBBQlub2RlSW5kZXgFDm5ld0xlYXNlQW1vdW50BQNuaWwFA25pbAEMcmVhZE5vZGVJbmZvAQdub2RlSWR4BAtub2RlQWRkcmVzcwkBHGdldFN0YWtpbmdOb2RlQWRkcmVzc0J5SW5kZXgBBQdub2RlSWR4BAxsZWFzZWRBbXRLRVkJARFnZXRMZWFzZUFtb3VudEtleQEFB25vZGVJZHgECWxlYXNlZEFtdAkBDmdldE51bWJlckJ5S2V5AQUMbGVhc2VkQW10S0VZBApsZWFzZUlkS0VZCQENZ2V0TGVhc2VJZEtleQEFB25vZGVJZHgEB2xlYXNlSWQJAQV2YWx1ZQEJAJwIAgUEdGhpcwUKbGVhc2VJZEtFWQkAlwoFBQtub2RlQWRkcmVzcwUMbGVhc2VkQW10S0VZBQlsZWFzZWRBbXQFCmxlYXNlSWRLRVkFB2xlYXNlSWQBCmNvbW1vblN3YXAFCHN3YXBUeXBlCXBtdEFtb3VudA51c2VyQWRkcmVzc1N0cgZ0eElkNTgbc3dhcFBhcmFtc0J5VXNlclNZU1JFQURPTkxZBA0kdDAxNzIyNDE3MzA0BRtzd2FwUGFyYW1zQnlVc2VyU1lTUkVBRE9OTFkEDHN3YXBMaW1pdE1heAgFDSR0MDE3MjI0MTczMDQCXzEEDnN3YXBMaW1pdFNwZW50CAUNJHQwMTcyMjQxNzMwNAJfMgQOYmxja3MyTG10UmVzZXQIBQ0kdDAxNzIyNDE3MzA0Al8zBA1taW5Td2FwQW1vdW50CQERbWluU3dhcEFtb3VudFJFQUQBBQhzd2FwVHlwZQQLdG90YWxMb2NrZWQJAQ90b3RhbExvY2tlZFJFQUQBBQhzd2FwVHlwZQQRdG90YWxMb2NrZWRCeVVzZXIJARV0b3RhbExvY2tlZEJ5VXNlclJFQUQCBQhzd2FwVHlwZQUOdXNlckFkZHJlc3NTdHIEC25vZGVBZGRyZXNzCQEVZ2V0U3Rha2luZ05vZGVCeUluZGV4AQAABAxwcmljZUJ5SW5kZXgJAQ9nZXRQcmljZUhpc3RvcnkBCQEVZ2V0SGVpZ2h0UHJpY2VCeUluZGV4AQUKcHJpY2VJbmRleAQMaXNTd2FwQnlOb2RlCQAAAgULbm9kZUFkZHJlc3MFDnVzZXJBZGRyZXNzU3RyBBZiYWxhbmNlTG9ja01heEludGVydmFsAwUMaXNTd2FwQnlOb2RlCQEbbm9kZUJhbGFuY2VMb2NrSW50ZXJ2YWxSRUFEAAkBF2JhbGFuY2VMb2NrSW50ZXJ2YWxSRUFEAQUIc3dhcFR5cGUEEHNlbGZVbmxvY2tIZWlnaHQJAGQCBQZoZWlnaHQFFmJhbGFuY2VMb2NrTWF4SW50ZXJ2YWwEDnN3YXBVc2RuVm9sdW1lAwkAAAIFCHN3YXBUeXBlAghuZXV0cmlubwUJcG10QW1vdW50CQEWY29udmVydFdhdmVzVG9OZXV0cmlubwIFCXBtdEFtb3VudAUMcHJpY2VCeUluZGV4AwkAZgIFDW1pblN3YXBBbW91bnQFCXBtdEFtb3VudAkBEW1pblN3YXBBbW91bnRGQUlMAgUIc3dhcFR5cGUFDW1pblN3YXBBbW91bnQDAwkBASEBBQxpc1N3YXBCeU5vZGUJAGYCBQ5zd2FwTGltaXRTcGVudAAABwkAAgEJAKwCAgI6WW91IGhhdmUgZXhjZWVkZWQgc3dhcCBsaW1pdCEgTmV4dCBhbGxvd2VkIHN3YXAgaGVpZ2h0IGlzIAkApAMBCQBkAgUGaGVpZ2h0BQ5ibGNrczJMbXRSZXNldAMDCQEBIQEFDGlzU3dhcEJ5Tm9kZQkAZgIFDnN3YXBVc2RuVm9sdW1lBQxzd2FwTGltaXRNYXgHCQACAQkArAICCQCsAgIJAKwCAgIuWW91IGhhdmUgZXhjZWVkZWQgeW91ciBzd2FwIGxpbWl0ISBSZXF1ZXN0ZWQ6IAkApAMBBQ5zd2FwVXNkblZvbHVtZQINLCBhdmFpbGFibGU6IAkApAMBBQxzd2FwTGltaXRNYXgDBQlpc0Jsb2NrZWQJARVlbWVyZ2VuY3lTaHV0ZG93bkZBSUwABAlsZWFzZVBhcnQDCQAAAgUIc3dhcFR5cGUCBXdhdmVzCQEWcHJlcGFyZVVubGVhc2VBbmRMZWFzZQEAAAUDbmlsCQCUCgIJAM4IAgkAzAgCCQEMSW50ZWdlckVudHJ5AgkBGGtleVN3YXBVc2VyU3BlbnRJblBlcmlvZAEFDnVzZXJBZGRyZXNzU3RyBQ5zd2FwVXNkblZvbHVtZQkAzAgCCQEMSW50ZWdlckVudHJ5AgkBFWtleVVzZXJMYXN0U3dhcEhlaWdodAEFDnVzZXJBZGRyZXNzU3RyBQZoZWlnaHQJAMwIAgkBDEludGVnZXJFbnRyeQIJARR0b3RhbExvY2tlZEJ5VXNlcktFWQIFCHN3YXBUeXBlBQ51c2VyQWRkcmVzc1N0cgkAZAIFEXRvdGFsTG9ja2VkQnlVc2VyBQlwbXRBbW91bnQJAMwIAgkBDEludGVnZXJFbnRyeQIJARhnZXRCYWxhbmNlVW5sb2NrQmxvY2tLZXkBBQ51c2VyQWRkcmVzc1N0cgUQc2VsZlVubG9ja0hlaWdodAkAzAgCCQEMSW50ZWdlckVudHJ5AgkBDnRvdGFsTG9ja2VkS0VZAQUIc3dhcFR5cGUJAGQCBQt0b3RhbExvY2tlZAUJcG10QW1vdW50CQDMCAIJAQtTdHJpbmdFbnRyeQIJAQdzd2FwS0VZAgUOdXNlckFkZHJlc3NTdHIFBnR4SWQ1OAkBD3BlbmRpbmdTd2FwREFUQQMFCHN3YXBUeXBlBQlwbXRBbW91bnQFEHNlbGZVbmxvY2tIZWlnaHQFA25pbAUJbGVhc2VQYXJ0BQR1bml0AQ5jb21tb25XaXRoZHJhdwQHYWNjb3VudAVpbmRleAhzd2FwVHhJZAx3aXRoZHJhd1R4SWQEC3VzZXJBZGRyZXNzCQERQGV4dHJOYXRpdmUoMTA2MikBBQdhY2NvdW50BAlkYXRhQXJyYXkJARJzd2FwRGF0YUZhaWxPclJFQUQCBQdhY2NvdW50BQhzd2FwVHhJZAQQc2VsZlVubG9ja0hlaWdodAkBDXBhcnNlSW50VmFsdWUBCQCRAwIFCWRhdGFBcnJheQUUc0lkeFNlbGZVbmxvY2tIZWlnaHQECHN3YXBUeXBlCQCRAwIFCWRhdGFBcnJheQUMc0lkeFN3YXBUeXBlBAhpbkFtb3VudAkBDXBhcnNlSW50VmFsdWUBCQCRAwIFCWRhdGFBcnJheQUMc0lkeEluQW1vdW50BApzd2FwU3RhdHVzCQCRAwIFCWRhdGFBcnJheQUKc0lkeFN0YXR1cwQLc3RhcnRIZWlnaHQJAQ1wYXJzZUludFZhbHVlAQkAkQMCBQlkYXRhQXJyYXkFD3NJZHhTdGFydEhlaWdodAQKb3V0RmVlUGFydAkBC3ZhbHVlT3JFbHNlAgkAmggCBQR0aGlzCQENb3V0RmVlUGFydEtFWQEFCHN3YXBUeXBlBQ5ERUZBVUxUU1dBUEZFRQQLdG90YWxMb2NrZWQJAQ90b3RhbExvY2tlZFJFQUQBBQhzd2FwVHlwZQQRdG90YWxMb2NrZWRCeVVzZXIJARV0b3RhbExvY2tlZEJ5VXNlclJFQUQCBQhzd2FwVHlwZQUHYWNjb3VudAQMdW5sb2NrSGVpZ2h0BRBzZWxmVW5sb2NrSGVpZ2h0BAtpbmRleEhlaWdodAkBFWdldEhlaWdodFByaWNlQnlJbmRleAEFBWluZGV4BA9wcmV2SW5kZXhIZWlnaHQJARVnZXRIZWlnaHRQcmljZUJ5SW5kZXgBCQBlAgUFaW5kZXgAAQQMcHJpY2VCeUluZGV4CQEPZ2V0UHJpY2VIaXN0b3J5AQULaW5kZXhIZWlnaHQEE291dEFtb3VudEdyb3NzVHVwbGUDCQAAAgUIc3dhcFR5cGUCBXdhdmVzCQCUCgIJARZjb252ZXJ0V2F2ZXNUb05ldXRyaW5vAgUIaW5BbW91bnQFDHByaWNlQnlJbmRleAUPbmV1dHJpbm9Bc3NldElkAwkAAAIFCHN3YXBUeXBlAghuZXV0cmlubwkAlAoCCQEWY29udmVydE5ldXRyaW5vVG9XYXZlcwIFCGluQW1vdW50BQxwcmljZUJ5SW5kZXgFBHVuaXQJAAIBCQCsAgICFlVuc3VwcG9ydGVkIHN3YXAgdHlwZSAFCHN3YXBUeXBlBAxwYXlvdXRzQXJyYXkJAQlhcHBseUZlZXMCCAUTb3V0QW1vdW50R3Jvc3NUdXBsZQJfMQUKb3V0RmVlUGFydAQMb3V0TmV0QW1vdW50CQCRAwIFDHBheW91dHNBcnJheQUMSWR4TmV0QW1vdW50BAxvdXRGZWVBbW91bnQJAJEDAgUMcGF5b3V0c0FycmF5BQxJZHhGZWVBbW91bnQDBQlpc0Jsb2NrZWQJARVlbWVyZ2VuY3lTaHV0ZG93bkZBSUwAAwkBAiE9AgUKc3dhcFN0YXR1cwIHUEVORElORwkAAgECH3N3YXAgaGFzIGJlZW4gYWxyZWFkeSBwcm9jZXNzZWQDCQBmAgUMdW5sb2NrSGVpZ2h0BQZoZWlnaHQJAAIBCQCsAgIJAKwCAgIRcGxlYXNlIHdhaXQgZm9yOiAJAKQDAQUMdW5sb2NrSGVpZ2h0Ah8gYmxvY2sgaGVpZ2h0IHRvIHdpdGhkcmF3IGZ1bmRzAwMDCQBmAgUFaW5kZXgFCnByaWNlSW5kZXgGCQBmAgUMdW5sb2NrSGVpZ2h0BQtpbmRleEhlaWdodAYDCQECIT0CBQ9wcmV2SW5kZXhIZWlnaHQAAAkAZwIFD3ByZXZJbmRleEhlaWdodAUMdW5sb2NrSGVpZ2h0BwkBDnByaWNlSW5kZXhGQUlMBQUFaW5kZXgFCnByaWNlSW5kZXgFC2luZGV4SGVpZ2h0BQx1bmxvY2tIZWlnaHQFD3ByZXZJbmRleEhlaWdodAMJAGcCAAAJAJEDAgUMcGF5b3V0c0FycmF5BQ5JZHhHcm9zc0Ftb3VudAkAAgECE2JhbGFuY2UgZXF1YWxzIHplcm8DAwkAZgIAAAUKb3V0RmVlUGFydAYJAGcCBQpvdXRGZWVQYXJ0BQVQQVVMSQkAAgEJAKwCAgkArAICCQCsAgICHmludmFsaWQgb3V0RmVlUGFydCBjb25maWcgZm9yIAUIc3dhcFR5cGUCEiBzd2FwOiBvdXRGZWVQYXJ0PQkApAMBBQpvdXRGZWVQYXJ0BA11bmxlYXNlQW1vdW50AwMJAAACBQhzd2FwVHlwZQIIbmV1dHJpbm8JAGYCCAUTb3V0QW1vdW50R3Jvc3NUdXBsZQJfMQAABwgFE291dEFtb3VudEdyb3NzVHVwbGUCXzEAAAQFc3RhdGUJAMwIAgkBDEludGVnZXJFbnRyeQIJARR0b3RhbExvY2tlZEJ5VXNlcktFWQIFCHN3YXBUeXBlBQdhY2NvdW50CQBlAgURdG90YWxMb2NrZWRCeVVzZXIFCGluQW1vdW50CQDMCAIJAQxJbnRlZ2VyRW50cnkCCQEOdG90YWxMb2NrZWRLRVkBBQhzd2FwVHlwZQkAZQIFC3RvdGFsTG9ja2VkBQhpbkFtb3VudAkAzAgCCQEOU2NyaXB0VHJhbnNmZXIDBQt1c2VyQWRkcmVzcwUMb3V0TmV0QW1vdW50CAUTb3V0QW1vdW50R3Jvc3NUdXBsZQJfMgkAzAgCCQELU3RyaW5nRW50cnkCCQEHc3dhcEtFWQIFB2FjY291bnQFCHN3YXBUeElkCQEOZmluaXNoU3dhcERBVEEHBQlkYXRhQXJyYXkFDHByaWNlQnlJbmRleAUMb3V0TmV0QW1vdW50BQxvdXRGZWVBbW91bnQFDHVubG9ja0hlaWdodAUFaW5kZXgFDHdpdGhkcmF3VHhJZAUDbmlsCQCVCgMFBXN0YXRlCQEPQXR0YWNoZWRQYXltZW50AggFE291dEFtb3VudEdyb3NzVHVwbGUCXzIFDG91dEZlZUFtb3VudAUNdW5sZWFzZUFtb3VudAwBaQELY29uc3RydWN0b3IMEm5ldXRyaW5vQXNzZXRJZFBybQ5ib25kQXNzZXRJZFBybRJhdWN0aW9uQ29udHJhY3RQcm0WbGlxdWlkYXRpb25Db250cmFjdFBybQ5ycGRDb250cmFjdFBybRtub2RlT3JhY2xlUHJvdmlkZXJQdWJLZXlQcm0bYmFsYW5jZVdhdmVzTG9ja0ludGVydmFsUHJtHmJhbGFuY2VOZXV0cmlub0xvY2tJbnRlcnZhbFBybRVtaW5XYXZlc1N3YXBBbW91bnRQcm0YbWluTmV1dHJpbm9Td2FwQW1vdW50UHJtFW5ldXRyaW5vT3V0RmVlUGFydFBybRJ3YXZlc091dEZlZVBhcnRQcm0EC2NoZWNrQ2FsbGVyCQEIdGhpc09ubHkBBQFpAwkAAAIFC2NoZWNrQ2FsbGVyBQtjaGVja0NhbGxlcgMJAQIhPQIJAJADAQgFAWkIcGF5bWVudHMAAAkAAgECE25vIHBheW1lbnRzIGFsbG93ZWQJAMwIAgkBC1N0cmluZ0VudHJ5AgUSTmV1dHJpbm9Bc3NldElkS2V5BRJuZXV0cmlub0Fzc2V0SWRQcm0JAMwIAgkBC1N0cmluZ0VudHJ5AgUOQm9uZEFzc2V0SWRLZXkFDmJvbmRBc3NldElkUHJtCQDMCAIJAQtTdHJpbmdFbnRyeQIFEkF1Y3Rpb25Db250cmFjdEtleQUSYXVjdGlvbkNvbnRyYWN0UHJtCQDMCAIJAQtTdHJpbmdFbnRyeQIFFkxpcXVpZGF0aW9uQ29udHJhY3RLZXkFFmxpcXVpZGF0aW9uQ29udHJhY3RQcm0JAMwIAgkBC1N0cmluZ0VudHJ5AgUOUlBEQ29udHJhY3RLZXkFDnJwZENvbnRyYWN0UHJtCQDMCAIJAQtTdHJpbmdFbnRyeQIFG05vZGVPcmFjbGVQcm92aWRlclB1YktleUtleQUbbm9kZU9yYWNsZVByb3ZpZGVyUHViS2V5UHJtCQDMCAIJAQxJbnRlZ2VyRW50cnkCBRtCYWxhbmNlV2F2ZXNMb2NrSW50ZXJ2YWxLZXkFG2JhbGFuY2VXYXZlc0xvY2tJbnRlcnZhbFBybQkAzAgCCQEMSW50ZWdlckVudHJ5AgUeQmFsYW5jZU5ldXRyaW5vTG9ja0ludGVydmFsS2V5BR5iYWxhbmNlTmV1dHJpbm9Mb2NrSW50ZXJ2YWxQcm0JAMwIAgkBDEludGVnZXJFbnRyeQIFFU1pbldhdmVzU3dhcEFtb3VudEtleQUVbWluV2F2ZXNTd2FwQW1vdW50UHJtCQDMCAIJAQxJbnRlZ2VyRW50cnkCBRhNaW5OZXV0cmlub1N3YXBBbW91bnRLZXkFGG1pbk5ldXRyaW5vU3dhcEFtb3VudFBybQkAzAgCCQEMSW50ZWdlckVudHJ5AgUVTmV1dHJpbm9PdXRGZWVQYXJ0S2V5BRVuZXV0cmlub091dEZlZVBhcnRQcm0JAMwIAgkBDEludGVnZXJFbnRyeQIFEldhdmVzT3V0RmVlUGFydEtleQUSd2F2ZXNPdXRGZWVQYXJ0UHJtBQNuaWwJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4BaQENY29uc3RydWN0b3JWMgMMbWF0aENvbnRyYWN0E25zYnRTdGFraW5nQ29udHJhY3QUc3dhcHNUaW1lZnJhbWVCbG9ja3MEC2NoZWNrQ2FsbGVyCQEIdGhpc09ubHkBBQFpAwkAAAIFC2NoZWNrQ2FsbGVyBQtjaGVja0NhbGxlcgMJAQIhPQIJAJADAQgFAWkIcGF5bWVudHMAAAkAAgECE25vIHBheW1lbnRzIGFsbG93ZWQJAMwIAgkBC1N0cmluZ0VudHJ5AgUPTWF0aENvbnRyYWN0S2V5BQxtYXRoQ29udHJhY3QJAMwIAgkBC1N0cmluZ0VudHJ5AgUWTnNidFN0YWtpbmdDb250cmFjdEtleQUTbnNidFN0YWtpbmdDb250cmFjdAkAzAgCCQEMSW50ZWdlckVudHJ5AgkBEXN3YXBzVGltZWZyYW1lS0VZAAUUc3dhcHNUaW1lZnJhbWVCbG9ja3MFA25pbAkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgFpARNzd2FwV2F2ZXNUb05ldXRyaW5vAAMJAQIhPQIJAJADAQgFAWkIcGF5bWVudHMAAQkAAgECLHN3YXBXYXZlc1RvTmV1dHJpbm8gcmVxdWlyZSBvbmx5IG9uZSBwYXltZW50BANwbXQJAQV2YWx1ZQEJAJEDAggFAWkIcGF5bWVudHMAAAMJAQlpc0RlZmluZWQBCAUDcG10B2Fzc2V0SWQJAAIBAilPbmx5IFdhdmVzIHRva2VuIGlzIGFsbG93ZWQgZm9yIHN3YXBwaW5nLgQLdXNlckFkZHJlc3MJAKUIAQgFAWkGY2FsbGVyBAZ0eElkNTgJANgEAQgFAWkNdHJhbnNhY3Rpb25JZAQQc3dhcFBhcmFtc1NUUlVDVAkBEmFzU3dhcFBhcmFtc1NUUlVDVAEJAPwHBAUEdGhpcwIbc3dhcFBhcmFtc0J5VXNlclNZU1JFQURPTkxZCQDMCAIFC3VzZXJBZGRyZXNzCQDMCAIAAAUDbmlsBQNuaWwEEGNvbW1vblN3YXBSZXN1bHQJAQpjb21tb25Td2FwBQIFd2F2ZXMIBQNwbXQGYW1vdW50BQt1c2VyQWRkcmVzcwUGdHhJZDU4BRBzd2FwUGFyYW1zU1RSVUNUBRBjb21tb25Td2FwUmVzdWx0AWkBE3N3YXBOZXV0cmlub1RvV2F2ZXMAAwkBAiE9AgkAkAMBCAUBaQhwYXltZW50cwABCQACAQIsc3dhcE5ldXRyaW5vVG9XYXZlcyByZXF1aXJlIG9ubHkgb25lIHBheW1lbnQEA3BtdAkBBXZhbHVlAQkAkQMCCAUBaQhwYXltZW50cwAAAwkBAiE9AggFA3BtdAdhc3NldElkBQ9uZXV0cmlub0Fzc2V0SWQJAAIBAjpPbmx5IGFwcHJvcHJpYXRlIE5ldXRyaW5vIHRva2VucyBhcmUgYWxsb3dlZCBmb3Igc3dhcHBpbmcuBAt1c2VyQWRkcmVzcwkApQgBCAUBaQZjYWxsZXIEBnR4SWQ1OAkA2AQBCAUBaQ10cmFuc2FjdGlvbklkBBBzd2FwUGFyYW1zU1RSVUNUCQESYXNTd2FwUGFyYW1zU1RSVUNUAQkA/AcEBQR0aGlzAhtzd2FwUGFyYW1zQnlVc2VyU1lTUkVBRE9OTFkJAMwIAgULdXNlckFkZHJlc3MJAMwIAgAABQNuaWwFA25pbAQQY29tbW9uU3dhcFJlc3VsdAkBCmNvbW1vblN3YXAFAghuZXV0cmlubwgFA3BtdAZhbW91bnQFC3VzZXJBZGRyZXNzBQZ0eElkNTgFEHN3YXBQYXJhbXNTVFJVQ1QFEGNvbW1vblN3YXBSZXN1bHQBaQEId2l0aGRyYXcDB2FjY291bnQFaW5kZXgIc3dhcFR4SWQEBHR4SWQJANgEAQgFAWkNdHJhbnNhY3Rpb25JZAMJAQIhPQIJAJADAQgFAWkIcGF5bWVudHMAAAkAAgECE25vIHBheW1lbnRzIGFsbG93ZWQEC2NvbW1vblR1cGxlCQEOY29tbW9uV2l0aGRyYXcEBQdhY2NvdW50BQVpbmRleAUIc3dhcFR4SWQFBHR4SWQEBXN0YXRlCAULY29tbW9uVHVwbGUCXzEEA2ZlZQgFC2NvbW1vblR1cGxlAl8yBAp1bmxlYXNlQW10CAULY29tbW9uVHVwbGUCXzMEEXVubGVhc2VJbnZPckVtcHR5CQD8BwQFBHRoaXMCF2ludGVybmFsVW5sZWFzZUFuZExlYXNlCQDMCAIFCnVubGVhc2VBbXQFA25pbAUDbmlsAwkAAAIFEXVubGVhc2VJbnZPckVtcHR5BRF1bmxlYXNlSW52T3JFbXB0eQQJZ25zYnREYXRhCQEJYXNBbnlMaXN0AQkA/AcEBRdnbnNidENvbnRyb2xsZXJDb250cmFjdAIUZ25zYnRJbmZvU1lTUkVBRE9OTFkJAMwIAgIABQNuaWwFA25pbAQNZ25zYnRBbXRUb3RhbAkBBWFzSW50AQkAkQMCBQlnbnNidERhdGEAAQQVZ25zYnRBbXRGcm9tU3VyZlRvdGFsCQEFYXNJbnQBCQCRAwIJAQlhc0FueUxpc3QBCQCRAwIFCWduc2J0RGF0YQADAAMECnN1cmZGZWVBbXQDCQECIT0CBQ1nbnNidEFtdFRvdGFsAAAJAGsDCAUDZmVlBmFtb3VudAUVZ25zYnRBbXRGcm9tU3VyZlRvdGFsBQ1nbnNidEFtdFRvdGFsAAAECm5zYnRGZWVBbXQJAGUCCAUDZmVlBmFtb3VudAUKc3VyZkZlZUFtdAQLc3VyZkRlcG9zaXQDCQBmAgUKc3VyZkZlZUFtdAAABAdzdXJmSW52CQD8BwQFE3N1cmZTdGFraW5nQ29udHJhY3QCB2RlcG9zaXQFA25pbAkAzAgCCQEPQXR0YWNoZWRQYXltZW50AggFA2ZlZQdhc3NldElkBQpzdXJmRmVlQW10BQNuaWwDCQAAAgUHc3VyZkludgUHc3VyZkludgUDbmlsCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuBQNuaWwDCQAAAgULc3VyZkRlcG9zaXQFC3N1cmZEZXBvc2l0BAtuc2J0RGVwb3NpdAMJAGYCBQpuc2J0RmVlQW10AAAEB25zYnRJbnYJAPwHBAUTbnNidFN0YWtpbmdDb250cmFjdAIHZGVwb3NpdAUDbmlsCQDMCAIJAQ9BdHRhY2hlZFBheW1lbnQCCAUDZmVlB2Fzc2V0SWQFCm5zYnRGZWVBbXQFA25pbAMJAAACBQduc2J0SW52BQduc2J0SW52BQNuaWwJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4FA25pbAMJAAACBQtuc2J0RGVwb3NpdAULbnNidERlcG9zaXQFBXN0YXRlCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuAWkBF2ludGVybmFsVW5sZWFzZUFuZExlYXNlAQ11bmxlYXNlQW1vdW50AwkBAiE9AggFAWkGY2FsbGVyBQR0aGlzCQACAQIsaW50ZXJuYWxVbmxlYXNlQW5kTGVhc2UgaXMgbm90IHB1YmxpYyBtZXRob2QJARZwcmVwYXJlVW5sZWFzZUFuZExlYXNlAQUNdW5sZWFzZUFtb3VudAFpARJ0cmFuc2ZlclVzZG5Ub1VzZXICBmFtb3VudARhZGRyAwkBAiE9AggFAWkGY2FsbGVyBQ9hdWN0aW9uQ29udHJhY3QJAAIBAiNPbmx5IGF1Y3Rpb24gY29udHJhY3QgaXMgYXV0aG9yaXplZAkAzAgCCQEOU2NyaXB0VHJhbnNmZXIDCQERQGV4dHJOYXRpdmUoMTA2MikBBQRhZGRyBQZhbW91bnQFD25ldXRyaW5vQXNzZXRJZAUDbmlsAWkBC2FjY2VwdFdhdmVzAAMJAQIhPQIIBQFpBmNhbGxlcgUPYXVjdGlvbkNvbnRyYWN0CQACAQIyQ3VycmVudGx5IG9ubHkgYXVjdGlvbiBjb250cmFjdCBpcyBhbGxvd2VkIHRvIGNhbGwJAJQKAgkBFnByZXBhcmVVbmxlYXNlQW5kTGVhc2UBAAACB3N1Y2Nlc3MBaQEPYXBwcm92ZUxlYXNpbmdzAgZuTGlzdFMIZ3JvdXBOdW0EBGxBbXQJAGgCADIFB1dBVkVMRVQEBW5JZHhzCQDMCAIAAAkAzAgCAAEJAMwIAgACCQDMCAIAAwkAzAgCAAQJAMwIAgAFCQDMCAIABgkAzAgCAAcFA25pbAQHbW5nUHViUwkBC3ZhbHVlT3JFbHNlAgkAoggBAhwlcyVzX19jZmdfX2xlYXNpbmdNYW5hZ2VyUHViAiw3QVVNWDU0dWtZTVl2UG1tYTd5b0ZmNU5qWmhzNEJ1NW56M0V6OUVWOHN1cgQGbW5nUHViCQDZBAEFB21uZ1B1YlMEDm5vZGVSZWdBZGRyU3RyCQELdmFsdWVPckVsc2UCCQCiCAECHyVzJXNfX2NmZ19fbm9kZXNSZWdpc3RyeUFkZHJlc3MCIzNQOXZLcVFLalVkbXBYQWZpV2F1OGtyUkVZQVkxWHI2OXBFBAtub2RlUmVnQWRkcgkBEUBleHRyTmF0aXZlKDEwNjIpAQUObm9kZVJlZ0FkZHJTdHIEEWxHcm91cE5vZGVMaXN0S0VZCQEYZ2V0TGVhc2VHcm91cE5vZGVMaXN0S2V5AQUIZ3JvdXBOdW0ECmxHck5vZGVPcHQJAJ0IAgUEdGhpcwURbEdyb3VwTm9kZUxpc3RLRVkDCQEJaXNEZWZpbmVkAQUKbEdyTm9kZU9wdAkAAgEJAKwCAgkArAICAgZncm91cCAJAKQDAQUIZ3JvdXBOdW0CFCBhbHJlYWR5IGluaXRpYWxpemVkBAVuTGlzdAkAtQkCBQZuTGlzdFMFA1NFUAQIZXhwQ291bnQJAJADAQUFbklkeHMDCQECIT0CCAUBaQ9jYWxsZXJQdWJsaWNLZXkFBm1uZ1B1YgkAAgECHmFwcHJvdmVMZWFzaW5ncyBub3QgYXV0aG9yaXplZAQNJHQwMjgzNzgyODQ0MAkBDHJlYWROb2RlSW5mbwEAAAQGbkFkZHIwCAUNJHQwMjgzNzgyODQ0MAJfMQQIbEFtdEtFWTAIBQ0kdDAyODM3ODI4NDQwAl8yBAVsQW10MAgFDSR0MDI4Mzc4Mjg0NDACXzMEB2xJZEtFWTAIBQ0kdDAyODM3ODI4NDQwAl80BARsSWQwCAUNJHQwMjgzNzgyODQ0MAJfNQQNJHQwMjg0NDMyODUwNQkBDHJlYWROb2RlSW5mbwEAAQQGbkFkZHIxCAUNJHQwMjg0NDMyODUwNQJfMQQIbEFtdEtFWTEIBQ0kdDAyODQ0MzI4NTA1Al8yBAVsQW10MQgFDSR0MDI4NDQzMjg1MDUCXzMEB2xJZEtFWTEIBQ0kdDAyODQ0MzI4NTA1Al80BARsSWQxCAUNJHQwMjg0NDMyODUwNQJfNQQFbmV3TDAJAMQIAgUGbkFkZHIwCQBlAgUFbEFtdDAJAGkCCQBoAgUEbEFtdAUIZXhwQ291bnQAAgQFbmV3TDEJAMQIAgUGbkFkZHIxCQBlAgUFbEFtdDEJAGkCCQBoAgUEbEFtdAUIZXhwQ291bnQAAgQKdmFsaWRhdGlvbgkA/AcEBQtub2RlUmVnQWRkcgIadmFsaWRhdGVBbmRBcHByb3ZlTGVhc2luZ3MJAMwIAgUGbkxpc3RTBQNuaWwFA25pbAMJAAACBQp2YWxpZGF0aW9uBQp2YWxpZGF0aW9uCgEjZm9yRWFjaE5vZGVWYWxpZGF0ZUFuZEdlbmVyYXRlTGVhc2UCAWEBaQQEbm9kZQkAkQMCBQVuTGlzdAUBaQQCbGEJAMQIAgkBEUBleHRyTmF0aXZlKDEwNjIpAQUEbm9kZQUEbEFtdAkAzggCBQFhCQDMCAIFAmxhCQDMCAIJAQtCaW5hcnlFbnRyeQIJARZnZXRMZWFzZUlkQnlBZGRyZXNzS2V5AQUEbm9kZQkBBWxjYWxjAQUCbGEJAMwIAgkBDEludGVnZXJFbnRyeQIJARpnZXRMZWFzZUFtb3VudEJ5QWRkcmVzc0tleQEFBG5vZGUFBGxBbXQFA25pbAkAzggCCQDMCAIJAQtTdHJpbmdFbnRyeQIFEWxHcm91cE5vZGVMaXN0S0VZBQZuTGlzdFMJAMwIAgkBC0JpbmFyeUVudHJ5AgUHbElkS0VZMAkBBWxjYWxjAQUFbmV3TDAJAMwIAgkBC0JpbmFyeUVudHJ5AgUHbElkS0VZMQkBBWxjYWxjAQUFbmV3TDEJAMwIAgkBDEludGVnZXJFbnRyeQIFCGxBbXRLRVkwCAUFbmV3TDAGYW1vdW50CQDMCAIJAQxJbnRlZ2VyRW50cnkCBQhsQW10S0VZMQgFBW5ld0wxBmFtb3VudAkAzAgCCQELTGVhc2VDYW5jZWwBBQRsSWQwCQDMCAIJAQtMZWFzZUNhbmNlbAEFBGxJZDEJAMwIAgUFbmV3TDAJAMwIAgUFbmV3TDEFA25pbAoAAiRsBQVuSWR4cwoAAiRzCQCQAwEFAiRsCgAFJGFjYzAFA25pbAoBBSRmMF8xAgIkYQIkaQMJAGcCBQIkaQUCJHMFAiRhCQEjZm9yRWFjaE5vZGVWYWxpZGF0ZUFuZEdlbmVyYXRlTGVhc2UCBQIkYQkAkQMCBQIkbAUCJGkKAQUkZjBfMgICJGECJGkDCQBnAgUCJGkFAiRzBQIkYQkAAgECE0xpc3Qgc2l6ZSBleGNlZWRzIDgJAQUkZjBfMgIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIFBSRhY2MwAAAAAQACAAMABAAFAAYABwAICQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuAWkBG3N3YXBQYXJhbXNCeVVzZXJTWVNSRUFET05MWQIOdXNlckFkZHJlc3NTdHIJZ25zYnREaWZmBAlnbnNidERhdGEJAQlhc0FueUxpc3QBCQD8BwQFF2duc2J0Q29udHJvbGxlckNvbnRyYWN0AhRnbnNidEluZm9TWVNSRUFET05MWQkAzAgCBQ51c2VyQWRkcmVzc1N0cgUDbmlsBQNuaWwECGduc2J0QW10CQBkAgkBBWFzSW50AQkAkQMCBQlnbnNidERhdGEAAAUJZ25zYnREaWZmBA1nbnNidEFtdFRvdGFsCQBkAgkBBWFzSW50AQkAkQMCBQlnbnNidERhdGEAAQUJZ25zYnREaWZmBAxzd2FwTGltaXRNYXgJAQVhc0ludAEJAPwHBAUMbWF0aENvbnRyYWN0AhVjYWxjU3dhcExpbWl0UkVBRE9OTFkJAMwIAgUIZ25zYnRBbXQFA25pbAUDbmlsBA5sYXN0U3dhcEhlaWdodAkBC3ZhbHVlT3JFbHNlAgkAmggCBQR0aGlzCQEVa2V5VXNlckxhc3RTd2FwSGVpZ2h0AQUOdXNlckFkZHJlc3NTdHIAAAQXc3dhcExpbWl0VGltZWxpZmVCbG9ja3MJARJzd2Fwc1RpbWVmcmFtZVJFQUQABBlwYXNzZWRCbG9ja3NBZnRlckxhc3RTd2FwCQBlAgUGaGVpZ2h0BQ5sYXN0U3dhcEhlaWdodAQRaXNTd2FwVGltZWxpZmVOZXcJAGcCBRlwYXNzZWRCbG9ja3NBZnRlckxhc3RTd2FwBRdzd2FwTGltaXRUaW1lbGlmZUJsb2NrcwQOc3dhcExpbWl0U3BlbnQDBRFpc1N3YXBUaW1lbGlmZU5ldwAACQELdmFsdWVPckVsc2UCCQCaCAIFBHRoaXMJARhrZXlTd2FwVXNlclNwZW50SW5QZXJpb2QBBQ51c2VyQWRkcmVzc1N0cgAABA5ibGNrczJMbXRSZXNldAMFEWlzU3dhcFRpbWVsaWZlTmV3AAAJAGUCBRdzd2FwTGltaXRUaW1lbGlmZUJsb2NrcwUZcGFzc2VkQmxvY2tzQWZ0ZXJMYXN0U3dhcAkAlAoCBQNuaWwJAJcKBQUMc3dhcExpbWl0TWF4BQ5zd2FwTGltaXRTcGVudAUOYmxja3MyTG10UmVzZXQFCGduc2J0QW10BQ1nbnNidEFtdFRvdGFsAWkBH3VwZGF0ZVJlc2VydmVzQW5kTmV1dHJpbm9TdXBwbHkACgEWZ2V0TnVtYmVyQnlLZXlJbnRlcm5hbAEDa2V5BAckbWF0Y2gwCQCaCAIFBHRoaXMFA2tleQMJAAECBQckbWF0Y2gwAgNJbnQEAWEFByRtYXRjaDAFAWEAAAQHbk1ldHJpeAkBCWFzQW55TGlzdAEJAPwHBAUMbWF0aENvbnRyYWN0AhpjYWxjTmV1dGlub01ldHJpY3NSRUFET05MWQUDbmlsBQNuaWwEA2lkeAkBFmdldE51bWJlckJ5S2V5SW50ZXJuYWwBAiJ1cGRhdGVSZXNlcnZlc0FuZE5ldXRyaW5vU3VwcGx5SWR4BAZuZXdJZHgJAGQCBQNpZHgAAQkAzAgCCQEMSW50ZWdlckVudHJ5AgIidXBkYXRlUmVzZXJ2ZXNBbmROZXV0cmlub1N1cHBseUlkeAUGbmV3SWR4CQDMCAIJAQxJbnRlZ2VyRW50cnkCAgdyZXNlcnZlCQEFYXNJbnQBCQCRAwIFB25NZXRyaXgAAwkAzAgCCQEMSW50ZWdlckVudHJ5AgIObmV1dHJpbm9TdXBwbHkJAQVhc0ludAEJAJEDAgUHbk1ldHJpeAAFCQDMCAIJAQxJbnRlZ2VyRW50cnkCAg5yZXNlcnZlc0luVXNkbgkBFmNvbnZlcnRXYXZlc1RvTmV1dHJpbm8CCQEFYXNJbnQBCQCRAwIFB25NZXRyaXgAAwkBBWFzSW50AQkAkQMCBQduTWV0cml4AAAJAMwIAgkBDEludGVnZXJFbnRyeQICB3N1cnBsdXMJAQVhc0ludAEJAJEDAgUHbk1ldHJpeAAGCQDMCAIJAQxJbnRlZ2VyRW50cnkCAgdkZWZpY2l0CQEBLQEJAQVhc0ludAEJAJEDAgUHbk1ldHJpeAAGBQNuaWwBaQEWd2F2ZXNCYWxhbmNlc1ZzUGF5bWVudAAEAWIJAO8HAQUEdGhpcwkAzAgCCQEMSW50ZWdlckVudHJ5AgIWd2F2ZXNCYWxhbmNlX2F2YWlsYWJsZQgFAWIJYXZhaWxhYmxlCQDMCAIJAQxJbnRlZ2VyRW50cnkCAhR3YXZlc0JhbGFuY2VfcmVndWxhcggFAWIHcmVndWxhcgkAzAgCCQEMSW50ZWdlckVudHJ5AgIXd2F2ZXNCYWxhbmNlX2dlbmVyYXRpbmcIBQFiCmdlbmVyYXRpbmcJAMwIAgkBDEludGVnZXJFbnRyeQICFndhdmVzQmFsYW5jZV9lZmZlY3RpdmUIBQFiCWVmZmVjdGl2ZQkAzAgCCQEMSW50ZWdlckVudHJ5AgINd2F2ZXNfcGF5bWVudAgJAQV2YWx1ZQEJAJEDAggFAWkIcGF5bWVudHMAAAZhbW91bnQFA25pbAECdHgBBnZlcmlmeQAEAmlkCQDYBAEIBQJ0eAJpZAQTcHViS2V5QWRtaW5zTGlzdFN0cgkAuQkCCQDMCAICLEV4dEVFSzE5bm1LajltQ3BuV3l2RUVKRllBVExNY1ZFTXZvaGhVSGt5SE5tCQDMCAICLEV2NXB5NUZmQlFYOWNacFlLbmZRclRCNDlCeWY4UW1wWldlRFZSaW00eVY3CQDMCAICLERVdXVMalh1OThuQndaYzdmcXdDVGp0QTNublJ3Z1Ria01TcjVTVTJObURSCQDMCAICLERVdXVMalh1OThuQndaYzdmcXdDVGp0QTNublJ3Z1Ria01TcjVTVTJObURSBQNuaWwFA1NFUAQQcHViS2V5QWRtaW5zTGlzdAkAtQkCCQELdmFsdWVPckVsc2UCCQCdCAIFD2NvbnRyb2xDb250cmFjdAIMJXNfX211bHRpc2lnBRNwdWJLZXlBZG1pbnNMaXN0U3RyBQNTRVAEBWNvdW50CQBkAgkAZAIJAGQCAwkA9AMDCAUCdHgJYm9keUJ5dGVzCQCRAwIIBQJ0eAZwcm9vZnMAAAkA2QQBCQCRAwIFEHB1YktleUFkbWluc0xpc3QAAAABAAADCQD0AwMIBQJ0eAlib2R5Qnl0ZXMJAJEDAggFAnR4BnByb29mcwABCQDZBAEJAJEDAgUQcHViS2V5QWRtaW5zTGlzdAABAAEAAAMJAPQDAwgFAnR4CWJvZHlCeXRlcwkAkQMCCAUCdHgGcHJvb2ZzAAIJANkEAQkAkQMCBRBwdWJLZXlBZG1pbnNMaXN0AAIAAQAAAwkA9AMDCAUCdHgJYm9keUJ5dGVzCQCRAwIIBQJ0eAZwcm9vZnMAAwkA2QQBCQCRAwIFEHB1YktleUFkbWluc0xpc3QAAwACAAAEByRtYXRjaDAFAnR4AwkAAQIFByRtYXRjaDACFVNwb25zb3JGZWVUcmFuc2FjdGlvbgQJc3BvbnNvclR4BQckbWF0Y2gwAwkBG2NoZWNrSXNWYWxpZE1pblNwb25zb3JlZEZlZQEFCXNwb25zb3JUeAkAZwIFBWNvdW50AAMHCQBnAgUFY291bnQAA6yzxMY=", "chainId": 84, "height": 2161025, "applicationStatus": "succeeded", "spentComplexity": 0 } View: original | compacted Prev: FtDNSChYhtAPm16gApha11hdpHjD7oyD4GUyieiUry2c Next: 5GeFHNn7ZzGPoCy4FJXBktMTYvd99GMQX6T7Za6zBjuZ Diff:
OldNewDifferences
1-{-# STDLIB_VERSION 5 #-}
1+{-# STDLIB_VERSION 6 #-}
22 {-# SCRIPT_TYPE ACCOUNT #-}
33 {-# CONTENT_TYPE DAPP #-}
4-func getStringOrFail (address,key) = valueOrErrorMessage(getString(address, key), (("Mandatory this." + key) + " is not defined"))
4+func getStringOrFail (address,key) = valueOrErrorMessage(getString(address, key), makeString(["mandatory ", toString(address), ".", key, " is not defined"], ""))
55
66
77 func lcalc (l) = calculateLeaseId(l)
1616 func getBoolByKey (key) = valueOrElse(getBoolean(this, key), false)
1717
1818
19-func getNumberByAddressAndKey (address,key) = valueOrElse(getInteger(addressFromStringValue(address), key), 0)
19+func getNumberByAddressAndKey (address,key) = valueOrElse(getInteger(address, key), 0)
2020
2121
2222 func getStringByAddressAndKey (address,key) = valueOrElse(getString(addressFromStringValue(address), key), "")
2323
2424
25-func getBoolByAddressAndKey (address,key) = valueOrElse(getBoolean(addressFromStringValue(address), key), false)
25+func getBoolByAddressAndKey (address,key) = valueOrElse(getBoolean(address, key), false)
2626
2727
2828 func asAnyList (v) = match v {
6565 }
6666
6767
68-let pubKeyAdminsList = ["ExtEEK19nmKj9mCpnWyvEEJFYATLMcVEMvohhUHkyHNm", "Ev5py5FfBQX9cZpYKnfQrTB49Byf8QmpZWeDVRim4yV7", "DUuuLjXu98nBwZc7fqwCTjtA3nnRwgTbkMSr5SU2NmDR", "5WRXFSjwcTbNfKcJs8ZqXmSSWYsSVJUtMvMqZj5hH4Nc"]
69-
7068 let SEP = "__"
7169
7270 let WAVELET = 100000000
8280 let IdxFeeAmount = 1
8381
8482 let IdxGrossAmount = 2
83+
84+let IdxControlCfgNeutrinoDapp = 1
85+
86+let IdxControlCfgAuctionDapp = 2
87+
88+let IdxControlCfgRpdDapp = 3
89+
90+let IdxControlCfgMathDapp = 4
91+
92+let IdxControlCfgLiquidationDapp = 5
93+
94+let IdxControlCfgRestDapp = 6
95+
96+let IdxControlCfgNodeRegistryDapp = 7
97+
98+let IdxControlCfgNsbtStakingDapp = 8
99+
100+let IdxControlCfgMediatorDapp = 9
101+
102+let IdxControlCfgSurfStakingDapp = 10
103+
104+let IdxControlCfgGnsbtControllerDapp = 11
105+
106+func keyControlAddress () = "%s%s__config__controlAddress"
107+
108+
109+func keyControlCfg () = "%s__controlConfig"
110+
111+
112+func readControlCfgOrFail (control) = split(getStringOrFail(control, keyControlCfg()), SEP)
113+
114+
115+func getContractAddressOrFail (controlCfg,idx) = valueOrErrorMessage(addressFromString(controlCfg[idx]), ("Control cfg doesn't contain address at index " + toString(idx)))
116+
117+
118+let controlContract = addressFromStringValue(valueOrElse(getString(this, keyControlAddress()), "3N4NS7d4Jo9a6F14LiFUKKYVdUkkf2eP4Zx"))
119+
120+let controlCfg = readControlCfgOrFail(controlContract)
121+
122+let mathContract = getContractAddressOrFail(controlCfg, IdxControlCfgMathDapp)
123+
124+let nsbtStakingContract = getContractAddressOrFail(controlCfg, IdxControlCfgNsbtStakingDapp)
125+
126+let surfStakingContract = getContractAddressOrFail(controlCfg, IdxControlCfgSurfStakingDapp)
127+
128+let gnsbtControllerContract = getContractAddressOrFail(controlCfg, IdxControlCfgGnsbtControllerDapp)
129+
130+let auctionContract = getContractAddressOrFail(controlCfg, IdxControlCfgAuctionDapp)
85131
86132 let NeutrinoAssetIdKey = "neutrino_asset_id"
87133
112158 let NeutrinoOutFeePartKey = "neutrinoOut_swap_feePart"
113159
114160 let WavesOutFeePartKey = "wavesOut_swap_feePart"
115-
116-let FeesManagerAddressKey = "fees_manager_address"
117161
118162 func keyNodeRegistry (address) = ("%s__" + address)
119163
202246 func keyUserLastSwapHeight (userAddress) = makeString(["%s%s", "userLastSwapHeight", userAddress], SEP)
203247
204248
205-func feeManagerAddressREAD () = valueOrErrorMessage(addressFromString(valueOrErrorMessage(getString(this, FeesManagerAddressKey), (FeesManagerAddressKey + " is not specified"))), (FeesManagerAddressKey + " invalid address format"))
206-
207-
208249 func convertNeutrinoToWaves (amount,price) = fraction(fraction(amount, PRICELET, price), WAVELET, PAULI)
209250
210251
226267 func priceIndexFAIL (index,priceIndex,indexHeight,unlockHeight,prevIndexHeight) = throw(((((((((("invalid price history index: index=" + toString(index)) + " priceIndex=") + toString(priceIndex)) + " indexHeight=") + toString(indexHeight)) + " unlockHeight=") + toString(unlockHeight)) + " prevIndexHeight=") + toString(prevIndexHeight)))
227268
228269
229-let liquidationContract = getStringByKey(LiquidationContractKey)
230-
231-let nsbtStakingContractStr = getStringByKey(NsbtStakingContractKey)
232-
233270 let neutrinoAssetId = fromBase58String(getStringByKey(NeutrinoAssetIdKey))
234-
235-let auctionContract = getStringByKey(AuctionContractKey)
236-
237-let rpdContract = getStringByKey(RPDContractKey)
238-
239-let controlContract = getStringByKey(ContolContractKey)
240-
241-let mathContractAddress = getStringByKey(MathContractKey)
242271
243272 let priceIndex = getNumberByAddressAndKey(controlContract, PriceIndexKey)
244273
246275
247276 let nodeOracleProviderPubKey = fromBase58String(getStringByKey(NodeOracleProviderPubKeyKey))
248277
249-let bondAssetId = fromBase58String("F3iaxzruFeKujfVfYSZEkejpjh67wmRfPCRHiNmWKp3Z")
278+let bondAssetId = fromBase58String("EyZUuypQp3WWhxqLNx8Y6N6GsbT21f5UCr36Xd7gsGCk")
250279
251280 let deprecatedBondAssetId = fromBase58String("975akZBfnMj513U7MZaHKzQrmsEx5aE3wdWKTrHBhbjF")
252281
253282 let neutrinoContract = this
254-
255-let mathContract = addressFromStringValue(mathContractAddress)
256-
257-let nsbtStakingContract = addressFromStringValue(nsbtStakingContractStr)
258283
259284 let currentPrice = getNumberByAddressAndKey(controlContract, PriceKey)
260285
398423
399424
400425 func commonSwap (swapType,pmtAmount,userAddressStr,txId58,swapParamsByUserSYSREADONLY) = {
401- let $t01675916839 = swapParamsByUserSYSREADONLY
402- let swapLimitMax = $t01675916839._1
403- let swapLimitSpent = $t01675916839._2
404- let blcks2LmtReset = $t01675916839._3
426+ let $t01722417304 = swapParamsByUserSYSREADONLY
427+ let swapLimitMax = $t01722417304._1
428+ let swapLimitSpent = $t01722417304._2
429+ let blcks2LmtReset = $t01722417304._3
405430 let minSwapAmount = minSwapAmountREAD(swapType)
406431 let totalLocked = totalLockedREAD(swapType)
407432 let totalLockedByUser = totalLockedByUserREAD(swapType, userAddressStr)
438463
439464 func commonWithdraw (account,index,swapTxId,withdrawTxId) = {
440465 let userAddress = addressFromStringValue(account)
441- let feeManagerAddress = feeManagerAddressREAD()
442466 let dataArray = swapDataFailOrREAD(account, swapTxId)
443467 let selfUnlockHeight = parseIntValue(dataArray[sIdxSelfUnlockHeight])
444468 let swapType = dataArray[sIdxSwapType]
564588 let unleaseAmt = commonTuple._3
565589 let unleaseInvOrEmpty = invoke(this, "internalUnleaseAndLease", [unleaseAmt], nil)
566590 if ((unleaseInvOrEmpty == unleaseInvOrEmpty))
567- then if ((0 >= fee.amount))
568- then state
569- else {
570- let nsbtStakingReward = invoke(nsbtStakingContract, "deposit", nil, [fee])
571- if ((nsbtStakingReward == nsbtStakingReward))
572- then state
573- else throw("Strict value is not equal to itself.")
574- }
591+ then {
592+ let gnsbtData = asAnyList(invoke(gnsbtControllerContract, "gnsbtInfoSYSREADONLY", [""], nil))
593+ let gnsbtAmtTotal = asInt(gnsbtData[1])
594+ let gnsbtAmtFromSurfTotal = asInt(asAnyList(gnsbtData[3])[3])
595+ let surfFeeAmt = if ((gnsbtAmtTotal != 0))
596+ then fraction(fee.amount, gnsbtAmtFromSurfTotal, gnsbtAmtTotal)
597+ else 0
598+ let nsbtFeeAmt = (fee.amount - surfFeeAmt)
599+ let surfDeposit = if ((surfFeeAmt > 0))
600+ then {
601+ let surfInv = invoke(surfStakingContract, "deposit", nil, [AttachedPayment(fee.assetId, surfFeeAmt)])
602+ if ((surfInv == surfInv))
603+ then nil
604+ else throw("Strict value is not equal to itself.")
605+ }
606+ else nil
607+ if ((surfDeposit == surfDeposit))
608+ then {
609+ let nsbtDeposit = if ((nsbtFeeAmt > 0))
610+ then {
611+ let nsbtInv = invoke(nsbtStakingContract, "deposit", nil, [AttachedPayment(fee.assetId, nsbtFeeAmt)])
612+ if ((nsbtInv == nsbtInv))
613+ then nil
614+ else throw("Strict value is not equal to itself.")
615+ }
616+ else nil
617+ if ((nsbtDeposit == nsbtDeposit))
618+ then state
619+ else throw("Strict value is not equal to itself.")
620+ }
621+ else throw("Strict value is not equal to itself.")
622+ }
575623 else throw("Strict value is not equal to itself.")
576624 }
577625 }
586634
587635
588636 @Callable(i)
589-func transferUsdnToUser (amount,addr) = if ((i.caller != addressFromStringValue(auctionContract)))
637+func transferUsdnToUser (amount,addr) = if ((i.caller != auctionContract))
590638 then throw("Only auction contract is authorized")
591639 else [ScriptTransfer(addressFromStringValue(addr), amount, neutrinoAssetId)]
592640
593641
594642
595643 @Callable(i)
596-func acceptWaves () = if ((i.caller != addressFromStringValue(auctionContract)))
644+func acceptWaves () = if ((i.caller != auctionContract))
597645 then throw("Currently only auction contract is allowed to call")
598646 else $Tuple2(prepareUnleaseAndLease(0), "success")
599647
601649
602650 @Callable(i)
603651 func approveLeasings (nListS,groupNum) = {
604- let lAmt = (500000 * WAVELET)
652+ let lAmt = (50 * WAVELET)
605653 let nIdxs = [0, 1, 2, 3, 4, 5, 6, 7]
606654 let mngPubS = valueOrElse(getString("%s%s__cfg__leasingManagerPub"), "7AUMX54ukYMYvPmma7yoFf5NjZhs4Bu5nz3Ez9EV8sur")
607655 let mngPub = fromBase58String(mngPubS)
617665 if ((i.callerPublicKey != mngPub))
618666 then throw("approveLeasings not authorized")
619667 else {
620- let $t02735227414 = readNodeInfo(0)
621- let nAddr0 = $t02735227414._1
622- let lAmtKEY0 = $t02735227414._2
623- let lAmt0 = $t02735227414._3
624- let lIdKEY0 = $t02735227414._4
625- let lId0 = $t02735227414._5
626- let $t02741727479 = readNodeInfo(1)
627- let nAddr1 = $t02741727479._1
628- let lAmtKEY1 = $t02741727479._2
629- let lAmt1 = $t02741727479._3
630- let lIdKEY1 = $t02741727479._4
631- let lId1 = $t02741727479._5
668+ let $t02837828440 = readNodeInfo(0)
669+ let nAddr0 = $t02837828440._1
670+ let lAmtKEY0 = $t02837828440._2
671+ let lAmt0 = $t02837828440._3
672+ let lIdKEY0 = $t02837828440._4
673+ let lId0 = $t02837828440._5
674+ let $t02844328505 = readNodeInfo(1)
675+ let nAddr1 = $t02844328505._1
676+ let lAmtKEY1 = $t02844328505._2
677+ let lAmt1 = $t02844328505._3
678+ let lIdKEY1 = $t02844328505._4
679+ let lId1 = $t02844328505._5
632680 let newL0 = Lease(nAddr0, (lAmt0 - ((lAmt * expCount) / 2)))
633681 let newL1 = Lease(nAddr1, (lAmt1 - ((lAmt * expCount) / 2)))
634682 let validation = invoke(nodeRegAddr, "validateAndApproveLeasings", [nListS], nil)
663711
664712
665713 @Callable(i)
666-func swapParamsByUserSYSREADONLY (userAddressStr,nsbtDiff) = {
667- let nsbtData = asAnyList(invoke(nsbtStakingContract, "nsbtStakingSYSREADONLY", [userAddressStr], nil))
668- if ((nsbtData == nsbtData))
669- then {
670- let gnsbtAmt = (asInt(nsbtData[0]) + nsbtDiff)
671- let gnsbtAmtTotal = (asInt(nsbtData[1]) + nsbtDiff)
672- let swapLimitMax = asInt(invoke(mathContract, "calcSwapLimitREADONLY", [gnsbtAmt], nil))
673- let lastSwapHeight = valueOrElse(getInteger(this, keyUserLastSwapHeight(userAddressStr)), 0)
674- let swapLimitTimelifeBlocks = swapsTimeframeREAD()
675- let passedBlocksAfterLastSwap = (height - lastSwapHeight)
676- let isSwapTimelifeNew = (passedBlocksAfterLastSwap >= swapLimitTimelifeBlocks)
677- let swapLimitSpent = if (isSwapTimelifeNew)
678- then 0
679- else valueOrElse(getInteger(this, keySwapUserSpentInPeriod(userAddressStr)), 0)
680- let blcks2LmtReset = if (isSwapTimelifeNew)
681- then 0
682- else (swapLimitTimelifeBlocks - passedBlocksAfterLastSwap)
683- $Tuple2(nil, $Tuple5(swapLimitMax, swapLimitSpent, blcks2LmtReset, gnsbtAmt, gnsbtAmtTotal))
684- }
685- else throw("Strict value is not equal to itself.")
714+func swapParamsByUserSYSREADONLY (userAddressStr,gnsbtDiff) = {
715+ let gnsbtData = asAnyList(invoke(gnsbtControllerContract, "gnsbtInfoSYSREADONLY", [userAddressStr], nil))
716+ let gnsbtAmt = (asInt(gnsbtData[0]) + gnsbtDiff)
717+ let gnsbtAmtTotal = (asInt(gnsbtData[1]) + gnsbtDiff)
718+ let swapLimitMax = asInt(invoke(mathContract, "calcSwapLimitREADONLY", [gnsbtAmt], nil))
719+ let lastSwapHeight = valueOrElse(getInteger(this, keyUserLastSwapHeight(userAddressStr)), 0)
720+ let swapLimitTimelifeBlocks = swapsTimeframeREAD()
721+ let passedBlocksAfterLastSwap = (height - lastSwapHeight)
722+ let isSwapTimelifeNew = (passedBlocksAfterLastSwap >= swapLimitTimelifeBlocks)
723+ let swapLimitSpent = if (isSwapTimelifeNew)
724+ then 0
725+ else valueOrElse(getInteger(this, keySwapUserSpentInPeriod(userAddressStr)), 0)
726+ let blcks2LmtReset = if (isSwapTimelifeNew)
727+ then 0
728+ else (swapLimitTimelifeBlocks - passedBlocksAfterLastSwap)
729+ $Tuple2(nil, $Tuple5(swapLimitMax, swapLimitSpent, blcks2LmtReset, gnsbtAmt, gnsbtAmtTotal))
730+ }
731+
732+
733+
734+@Callable(i)
735+func updateReservesAndNeutrinoSupply () = {
736+ func getNumberByKeyInternal (key) = match getInteger(this, key) {
737+ case a: Int =>
738+ a
739+ case _ =>
740+ 0
741+ }
742+
743+ let nMetrix = asAnyList(invoke(mathContract, "calcNeutinoMetricsREADONLY", nil, nil))
744+ let idx = getNumberByKeyInternal("updateReservesAndNeutrinoSupplyIdx")
745+ let newIdx = (idx + 1)
746+[IntegerEntry("updateReservesAndNeutrinoSupplyIdx", newIdx), IntegerEntry("reserve", asInt(nMetrix[3])), IntegerEntry("neutrinoSupply", asInt(nMetrix[5])), IntegerEntry("reservesInUsdn", convertWavesToNeutrino(asInt(nMetrix[3]), asInt(nMetrix[0]))), IntegerEntry("surplus", asInt(nMetrix[6])), IntegerEntry("deficit", -(asInt(nMetrix[6])))]
747+ }
748+
749+
750+
751+@Callable(i)
752+func wavesBalancesVsPayment () = {
753+ let b = wavesBalance(this)
754+[IntegerEntry("wavesBalance_available", b.available), IntegerEntry("wavesBalance_regular", b.regular), IntegerEntry("wavesBalance_generating", b.generating), IntegerEntry("wavesBalance_effective", b.effective), IntegerEntry("waves_payment", value(i.payments[0]).amount)]
686755 }
687756
688757
689758 @Verifier(tx)
690759 func verify () = {
691760 let id = toBase58String(tx.id)
761+ let pubKeyAdminsListStr = makeString(["ExtEEK19nmKj9mCpnWyvEEJFYATLMcVEMvohhUHkyHNm", "Ev5py5FfBQX9cZpYKnfQrTB49Byf8QmpZWeDVRim4yV7", "DUuuLjXu98nBwZc7fqwCTjtA3nnRwgTbkMSr5SU2NmDR", "DUuuLjXu98nBwZc7fqwCTjtA3nnRwgTbkMSr5SU2NmDR"], SEP)
762+ let pubKeyAdminsList = split(valueOrElse(getString(controlContract, "%s__multisig"), pubKeyAdminsListStr), SEP)
692763 let count = ((((if (sigVerify(tx.bodyBytes, tx.proofs[0], fromBase58String(pubKeyAdminsList[0])))
693764 then 1
694765 else 0) + (if (sigVerify(tx.bodyBytes, tx.proofs[1], fromBase58String(pubKeyAdminsList[1])))
Full:
OldNewDifferences
1-{-# STDLIB_VERSION 5 #-}
1+{-# STDLIB_VERSION 6 #-}
22 {-# SCRIPT_TYPE ACCOUNT #-}
33 {-# CONTENT_TYPE DAPP #-}
4-func getStringOrFail (address,key) = valueOrErrorMessage(getString(address, key), (("Mandatory this." + key) + " is not defined"))
4+func getStringOrFail (address,key) = valueOrErrorMessage(getString(address, key), makeString(["mandatory ", toString(address), ".", key, " is not defined"], ""))
55
66
77 func lcalc (l) = calculateLeaseId(l)
88
99
1010 func getNumberByKey (key) = valueOrElse(getInteger(this, key), 0)
1111
1212
1313 func getStringByKey (key) = valueOrElse(getString(this, key), "")
1414
1515
1616 func getBoolByKey (key) = valueOrElse(getBoolean(this, key), false)
1717
1818
19-func getNumberByAddressAndKey (address,key) = valueOrElse(getInteger(addressFromStringValue(address), key), 0)
19+func getNumberByAddressAndKey (address,key) = valueOrElse(getInteger(address, key), 0)
2020
2121
2222 func getStringByAddressAndKey (address,key) = valueOrElse(getString(addressFromStringValue(address), key), "")
2323
2424
25-func getBoolByAddressAndKey (address,key) = valueOrElse(getBoolean(addressFromStringValue(address), key), false)
25+func getBoolByAddressAndKey (address,key) = valueOrElse(getBoolean(address, key), false)
2626
2727
2828 func asAnyList (v) = match v {
2929 case l: List[Any] =>
3030 l
3131 case _ =>
3232 throw("fail to cast into List[Any]")
3333 }
3434
3535
3636 func asString (v) = match v {
3737 case s: String =>
3838 s
3939 case _ =>
4040 throw("fail to cast into String")
4141 }
4242
4343
4444 func asInt (v) = match v {
4545 case i: Int =>
4646 i
4747 case _ =>
4848 throw("fail to cast into Int")
4949 }
5050
5151
5252 func asPayment (v) = match v {
5353 case p: AttachedPayment =>
5454 p
5555 case _ =>
5656 throw("fail to cast into AttachedPayment")
5757 }
5858
5959
6060 func asSwapParamsSTRUCT (v) = match v {
6161 case struct: (Int, Int, Int, Int, Int) =>
6262 struct
6363 case _ =>
6464 throw("fail to cast into Int")
6565 }
6666
6767
68-let pubKeyAdminsList = ["ExtEEK19nmKj9mCpnWyvEEJFYATLMcVEMvohhUHkyHNm", "Ev5py5FfBQX9cZpYKnfQrTB49Byf8QmpZWeDVRim4yV7", "DUuuLjXu98nBwZc7fqwCTjtA3nnRwgTbkMSr5SU2NmDR", "5WRXFSjwcTbNfKcJs8ZqXmSSWYsSVJUtMvMqZj5hH4Nc"]
69-
7068 let SEP = "__"
7169
7270 let WAVELET = 100000000
7371
7472 let PAULI = 1000000
7573
7674 let PRICELET = 1000000
7775
7876 let DEFAULTSWAPFEE = 20000
7977
8078 let IdxNetAmount = 0
8179
8280 let IdxFeeAmount = 1
8381
8482 let IdxGrossAmount = 2
83+
84+let IdxControlCfgNeutrinoDapp = 1
85+
86+let IdxControlCfgAuctionDapp = 2
87+
88+let IdxControlCfgRpdDapp = 3
89+
90+let IdxControlCfgMathDapp = 4
91+
92+let IdxControlCfgLiquidationDapp = 5
93+
94+let IdxControlCfgRestDapp = 6
95+
96+let IdxControlCfgNodeRegistryDapp = 7
97+
98+let IdxControlCfgNsbtStakingDapp = 8
99+
100+let IdxControlCfgMediatorDapp = 9
101+
102+let IdxControlCfgSurfStakingDapp = 10
103+
104+let IdxControlCfgGnsbtControllerDapp = 11
105+
106+func keyControlAddress () = "%s%s__config__controlAddress"
107+
108+
109+func keyControlCfg () = "%s__controlConfig"
110+
111+
112+func readControlCfgOrFail (control) = split(getStringOrFail(control, keyControlCfg()), SEP)
113+
114+
115+func getContractAddressOrFail (controlCfg,idx) = valueOrErrorMessage(addressFromString(controlCfg[idx]), ("Control cfg doesn't contain address at index " + toString(idx)))
116+
117+
118+let controlContract = addressFromStringValue(valueOrElse(getString(this, keyControlAddress()), "3N4NS7d4Jo9a6F14LiFUKKYVdUkkf2eP4Zx"))
119+
120+let controlCfg = readControlCfgOrFail(controlContract)
121+
122+let mathContract = getContractAddressOrFail(controlCfg, IdxControlCfgMathDapp)
123+
124+let nsbtStakingContract = getContractAddressOrFail(controlCfg, IdxControlCfgNsbtStakingDapp)
125+
126+let surfStakingContract = getContractAddressOrFail(controlCfg, IdxControlCfgSurfStakingDapp)
127+
128+let gnsbtControllerContract = getContractAddressOrFail(controlCfg, IdxControlCfgGnsbtControllerDapp)
129+
130+let auctionContract = getContractAddressOrFail(controlCfg, IdxControlCfgAuctionDapp)
85131
86132 let NeutrinoAssetIdKey = "neutrino_asset_id"
87133
88134 let BondAssetIdKey = "bond_asset_id"
89135
90136 let AuctionContractKey = "auction_contract"
91137
92138 let NsbtStakingContractKey = "nsbtStakingContract"
93139
94140 let LiquidationContractKey = "liquidation_contract"
95141
96142 let RPDContractKey = "rpd_contract"
97143
98144 let ContolContractKey = "control_contract"
99145
100146 let MathContractKey = "math_contract"
101147
102148 let BalanceWavesLockIntervalKey = "balance_waves_lock_interval"
103149
104150 let BalanceNeutrinoLockIntervalKey = "balance_neutrino_lock_interval"
105151
106152 let MinWavesSwapAmountKey = "min_waves_swap_amount"
107153
108154 let MinNeutrinoSwapAmountKey = "min_neutrino_swap_amount"
109155
110156 let NodeOracleProviderPubKeyKey = "node_oracle_provider"
111157
112158 let NeutrinoOutFeePartKey = "neutrinoOut_swap_feePart"
113159
114160 let WavesOutFeePartKey = "wavesOut_swap_feePart"
115-
116-let FeesManagerAddressKey = "fees_manager_address"
117161
118162 func keyNodeRegistry (address) = ("%s__" + address)
119163
120164
121165 let PriceKey = "price"
122166
123167 let PriceIndexKey = "price_index"
124168
125169 let IsBlockedKey = "is_blocked"
126170
127171 func getPriceHistoryKey (block) = ((PriceKey + "_") + toString(block))
128172
129173
130174 func getHeightPriceByIndexKey (index) = ((PriceIndexKey + "_") + toString(index))
131175
132176
133177 func getStakingNodeByIndex (idx) = getStringByKey(makeString(["%s%d%s", "lease", toString(idx), "nodeAddress"], SEP))
134178
135179
136180 func getStakingNodeAddressByIndex (idx) = addressFromStringValue(getStakingNodeByIndex(idx))
137181
138182
139183 func getReservedAmountForSponsorship () = valueOrElse(getInteger(this, makeString(["%s%s", "lease", "sponsorshipWavesReserve"], SEP)), (1000 * WAVELET))
140184
141185
142186 func getBalanceUnlockBlockKey (owner) = ("balance_unlock_block_" + owner)
143187
144188
145189 func getLeaseIdKey (nodeIndex) = makeString(["%s%d%s", "lease", toString(nodeIndex), "id"], SEP)
146190
147191
148192 func getLeaseIdByAddressKey (nodeAddress) = makeString(["%s%s%s", "leaseByAddress", nodeAddress, "id"], SEP)
149193
150194
151195 func getLeaseAmountKey (nodeIndex) = makeString(["%s%d%s", "lease", toString(nodeIndex), "amount"], SEP)
152196
153197
154198 func getLeaseAmountByAddressKey (nodeAddress) = makeString(["%s%s%s", "leaseByAddress", nodeAddress, "amount"], SEP)
155199
156200
157201 func getLeaseGroupNodeListKey (groupNum) = makeString(["%s%d%s", "leaseGroup", toString(groupNum), "nodeList"], SEP)
158202
159203
160204 func minSwapAmountKEY (swapType) = (("min_" + swapType) + "_swap_amount")
161205
162206
163207 func totalLockedKEY (swapType) = ("balance_lock_" + swapType)
164208
165209
166210 func totalLockedByUserKEY (swapType,owner) = makeString(["balance_lock", swapType, owner], "_")
167211
168212
169213 func balanceLockIntervalKEY (swapType) = (("balance_" + swapType) + "_lock_interval")
170214
171215
172216 func nodeBalanceLockIntervalKEY () = "balance_node_lock_interval"
173217
174218
175219 func outFeePartKEY (swapType) = (swapType + "Out_swap_feePart")
176220
177221
178222 func swapsTimeframeKEY () = "swaps_timeframe"
179223
180224
181225 func minSwapAmountREAD (swapType) = valueOrElse(getInteger(this, minSwapAmountKEY(swapType)), 0)
182226
183227
184228 func swapsTimeframeREAD () = valueOrElse(getInteger(this, swapsTimeframeKEY()), 1440)
185229
186230
187231 func totalLockedREAD (swapType) = valueOrElse(getInteger(this, totalLockedKEY(swapType)), 0)
188232
189233
190234 func totalLockedByUserREAD (swapType,owner) = valueOrElse(getInteger(this, totalLockedByUserKEY(swapType, owner)), 0)
191235
192236
193237 func balanceLockIntervalREAD (swapType) = valueOrElse(getInteger(this, balanceLockIntervalKEY(swapType)), 1440)
194238
195239
196240 func nodeBalanceLockIntervalREAD () = valueOrElse(getInteger(this, nodeBalanceLockIntervalKEY()), 1)
197241
198242
199243 func keySwapUserSpentInPeriod (userAddress) = makeString(["%s%s", "swapUserSpentInPeriod", userAddress], SEP)
200244
201245
202246 func keyUserLastSwapHeight (userAddress) = makeString(["%s%s", "userLastSwapHeight", userAddress], SEP)
203247
204248
205-func feeManagerAddressREAD () = valueOrErrorMessage(addressFromString(valueOrErrorMessage(getString(this, FeesManagerAddressKey), (FeesManagerAddressKey + " is not specified"))), (FeesManagerAddressKey + " invalid address format"))
206-
207-
208249 func convertNeutrinoToWaves (amount,price) = fraction(fraction(amount, PRICELET, price), WAVELET, PAULI)
209250
210251
211252 func convertWavesToNeutrino (amount,price) = fraction(fraction(amount, price, PRICELET), PAULI, WAVELET)
212253
213254
214255 func convertWavesToBond (amount,price) = convertWavesToNeutrino(amount, price)
215256
216257
217258 func convertJsonArrayToList (jsonArray) = split(jsonArray, ",")
218259
219260
220261 func minSwapAmountFAIL (swapType,minSwapAmount) = throw(((("The specified amount in " + swapType) + " swap is less than the required minimum of ") + toString(minSwapAmount)))
221262
222263
223264 func emergencyShutdownFAIL () = throw("contract is blocked by EMERGENCY SHUTDOWN actions untill reactivation by emergency oracles")
224265
225266
226267 func priceIndexFAIL (index,priceIndex,indexHeight,unlockHeight,prevIndexHeight) = throw(((((((((("invalid price history index: index=" + toString(index)) + " priceIndex=") + toString(priceIndex)) + " indexHeight=") + toString(indexHeight)) + " unlockHeight=") + toString(unlockHeight)) + " prevIndexHeight=") + toString(prevIndexHeight)))
227268
228269
229-let liquidationContract = getStringByKey(LiquidationContractKey)
230-
231-let nsbtStakingContractStr = getStringByKey(NsbtStakingContractKey)
232-
233270 let neutrinoAssetId = fromBase58String(getStringByKey(NeutrinoAssetIdKey))
234-
235-let auctionContract = getStringByKey(AuctionContractKey)
236-
237-let rpdContract = getStringByKey(RPDContractKey)
238-
239-let controlContract = getStringByKey(ContolContractKey)
240-
241-let mathContractAddress = getStringByKey(MathContractKey)
242271
243272 let priceIndex = getNumberByAddressAndKey(controlContract, PriceIndexKey)
244273
245274 let isBlocked = getBoolByAddressAndKey(controlContract, IsBlockedKey)
246275
247276 let nodeOracleProviderPubKey = fromBase58String(getStringByKey(NodeOracleProviderPubKeyKey))
248277
249-let bondAssetId = fromBase58String("F3iaxzruFeKujfVfYSZEkejpjh67wmRfPCRHiNmWKp3Z")
278+let bondAssetId = fromBase58String("EyZUuypQp3WWhxqLNx8Y6N6GsbT21f5UCr36Xd7gsGCk")
250279
251280 let deprecatedBondAssetId = fromBase58String("975akZBfnMj513U7MZaHKzQrmsEx5aE3wdWKTrHBhbjF")
252281
253282 let neutrinoContract = this
254-
255-let mathContract = addressFromStringValue(mathContractAddress)
256-
257-let nsbtStakingContract = addressFromStringValue(nsbtStakingContractStr)
258283
259284 let currentPrice = getNumberByAddressAndKey(controlContract, PriceKey)
260285
261286 func checkIsValidMinSponsoredFee (tx) = {
262287 let MINTRANSFERFEE = 100000
263288 let SponsoredFeeUpperBound = 1000
264289 let realNeutrinoFee = convertWavesToNeutrino(MINTRANSFERFEE, currentPrice)
265290 let minNeutrinoFee = (realNeutrinoFee * 2)
266291 let maxNeutrinoFee = fraction(realNeutrinoFee, SponsoredFeeUpperBound, 100)
267292 let inputFee = value(tx.minSponsoredAssetFee)
268293 if (if ((inputFee >= minNeutrinoFee))
269294 then (maxNeutrinoFee >= inputFee)
270295 else false)
271296 then (tx.assetId == neutrinoAssetId)
272297 else false
273298 }
274299
275300
276301 func getPriceHistory (block) = getNumberByAddressAndKey(controlContract, getPriceHistoryKey(block))
277302
278303
279304 func getHeightPriceByIndex (index) = getNumberByAddressAndKey(controlContract, getHeightPriceByIndexKey(index))
280305
281306
282307 func keyLockParamUserAmount (userAddress) = makeString(["%s%s%s", "paramByUser", userAddress, "amount"], SEP)
283308
284309
285310 let sIdxSwapType = 1
286311
287312 let sIdxStatus = 2
288313
289314 let sIdxInAmount = 3
290315
291316 let sIdxPrice = 4
292317
293318 let sIdxOutNetAmount = 5
294319
295320 let sIdxOutFeeAmount = 6
296321
297322 let sIdxStartHeight = 7
298323
299324 let sIdxStartTimestamp = 8
300325
301326 let sIdxEndHeight = 9
302327
303328 let sIdxEndTimestamp = 10
304329
305330 let sIdxSelfUnlockHeight = 11
306331
307332 let sIdxRandUnlockHeight = 12
308333
309334 let sIdxIndex = 13
310335
311336 let sIdxWithdrawTxId = 14
312337
313338 let sIdxMinRand = 15
314339
315340 let sIdxMaxRand = 16
316341
317342 func swapKEY (userAddress,txId) = makeString(["%s%s", userAddress, txId], SEP)
318343
319344
320345 func strSwapDATA (swapType,status,inAmount,price,outNetAmount,outFeeAmount,startHeight,startTimestamp,endHeight,endTimestamp,selfUnlockHeight,randUnlockHeight,index,withdrawTxId,randMin,randMax) = makeString(["%s%s%d%d%d%d%d%d%d%d%d%d%d%s", swapType, status, inAmount, price, outNetAmount, outFeeAmount, startHeight, startTimestamp, endHeight, endTimestamp, selfUnlockHeight, randUnlockHeight, index, withdrawTxId, randMin, randMax], SEP)
321346
322347
323348 func pendingSwapDATA (swapType,inAssetAmount,selfUnlockHeight) = strSwapDATA(swapType, "PENDING", toString(inAssetAmount), "0", "0", "0", toString(height), toString(lastBlock.timestamp), "0", "0", toString(selfUnlockHeight), "0", "0", "NULL", "0", "0")
324349
325350
326351 func finishSwapDATA (dataArray,price,outNetAmount,outFeeAmount,randUnlockHeight,index,withdrawTxId) = strSwapDATA(dataArray[sIdxSwapType], "FINISHED", dataArray[sIdxInAmount], toString(price), toString(outNetAmount), toString(outFeeAmount), dataArray[sIdxStartHeight], dataArray[sIdxStartTimestamp], toString(height), toString(lastBlock.timestamp), dataArray[sIdxSelfUnlockHeight], toString(randUnlockHeight), toString(index), withdrawTxId, dataArray[sIdxMinRand], dataArray[sIdxMaxRand])
327352
328353
329354 func swapDataFailOrREAD (userAddress,swapTxId) = {
330355 let swapKey = swapKEY(userAddress, swapTxId)
331356 split(valueOrErrorMessage(getString(this, swapKey), ("no swap data for " + swapKey)), SEP)
332357 }
333358
334359
335360 func applyFees (amountGross,feePart) = {
336361 let feeAmount = fraction(amountGross, feePart, PAULI)
337362 [(amountGross - feeAmount), feeAmount, amountGross]
338363 }
339364
340365
341366 func abs (x) = if ((0 > x))
342367 then -(x)
343368 else x
344369
345370
346371 func selectNode (unleaseAmount) = {
347372 let amountToLease = ((wavesBalance(neutrinoContract).available - unleaseAmount) - getReservedAmountForSponsorship())
348373 let oldLeased0 = getNumberByKey(getLeaseAmountKey(0))
349374 let oldLeased1 = getNumberByKey(getLeaseAmountKey(1))
350375 let newLeased0 = (amountToLease + oldLeased0)
351376 let newLeased1 = (amountToLease + oldLeased1)
352377 if (if ((newLeased0 > 0))
353378 then true
354379 else (newLeased1 > 0))
355380 then {
356381 let delta0 = abs((newLeased0 - oldLeased1))
357382 let delta1 = abs((newLeased1 - oldLeased0))
358383 if ((delta1 >= delta0))
359384 then $Tuple2(0, newLeased0)
360385 else $Tuple2(1, newLeased1)
361386 }
362387 else $Tuple2(-1, 0)
363388 }
364389
365390
366391 func thisOnly (i) = if ((i.caller != this))
367392 then throw("Permission denied: this contract only allowed")
368393 else true
369394
370395
371396 func prepareUnleaseAndLease (unleaseAmount) = {
372397 let nodeTuple = selectNode(unleaseAmount)
373398 let nodeIndex = nodeTuple._1
374399 let newLeaseAmount = nodeTuple._2
375400 if ((newLeaseAmount > 0))
376401 then {
377402 let leaseIdKey = getLeaseIdKey(nodeIndex)
378403 let oldLease = getBinary(this, leaseIdKey)
379404 let unleaseOrEmpty = if (isDefined(oldLease))
380405 then [LeaseCancel(value(oldLease))]
381406 else nil
382407 let leaseAmountKey = getLeaseAmountKey(nodeIndex)
383408 let lease = Lease(getStakingNodeAddressByIndex(nodeIndex), newLeaseAmount)
384409 (unleaseOrEmpty ++ [lease, BinaryEntry(leaseIdKey, lcalc(lease)), IntegerEntry(getLeaseAmountKey(nodeIndex), newLeaseAmount)])
385410 }
386411 else nil
387412 }
388413
389414
390415 func readNodeInfo (nodeIdx) = {
391416 let nodeAddress = getStakingNodeAddressByIndex(nodeIdx)
392417 let leasedAmtKEY = getLeaseAmountKey(nodeIdx)
393418 let leasedAmt = getNumberByKey(leasedAmtKEY)
394419 let leaseIdKEY = getLeaseIdKey(nodeIdx)
395420 let leaseId = value(getBinary(this, leaseIdKEY))
396421 $Tuple5(nodeAddress, leasedAmtKEY, leasedAmt, leaseIdKEY, leaseId)
397422 }
398423
399424
400425 func commonSwap (swapType,pmtAmount,userAddressStr,txId58,swapParamsByUserSYSREADONLY) = {
401- let $t01675916839 = swapParamsByUserSYSREADONLY
402- let swapLimitMax = $t01675916839._1
403- let swapLimitSpent = $t01675916839._2
404- let blcks2LmtReset = $t01675916839._3
426+ let $t01722417304 = swapParamsByUserSYSREADONLY
427+ let swapLimitMax = $t01722417304._1
428+ let swapLimitSpent = $t01722417304._2
429+ let blcks2LmtReset = $t01722417304._3
405430 let minSwapAmount = minSwapAmountREAD(swapType)
406431 let totalLocked = totalLockedREAD(swapType)
407432 let totalLockedByUser = totalLockedByUserREAD(swapType, userAddressStr)
408433 let nodeAddress = getStakingNodeByIndex(0)
409434 let priceByIndex = getPriceHistory(getHeightPriceByIndex(priceIndex))
410435 let isSwapByNode = (nodeAddress == userAddressStr)
411436 let balanceLockMaxInterval = if (isSwapByNode)
412437 then nodeBalanceLockIntervalREAD()
413438 else balanceLockIntervalREAD(swapType)
414439 let selfUnlockHeight = (height + balanceLockMaxInterval)
415440 let swapUsdnVolume = if ((swapType == "neutrino"))
416441 then pmtAmount
417442 else convertWavesToNeutrino(pmtAmount, priceByIndex)
418443 if ((minSwapAmount > pmtAmount))
419444 then minSwapAmountFAIL(swapType, minSwapAmount)
420445 else if (if (!(isSwapByNode))
421446 then (swapLimitSpent > 0)
422447 else false)
423448 then throw(("You have exceeded swap limit! Next allowed swap height is " + toString((height + blcks2LmtReset))))
424449 else if (if (!(isSwapByNode))
425450 then (swapUsdnVolume > swapLimitMax)
426451 else false)
427452 then throw(((("You have exceeded your swap limit! Requested: " + toString(swapUsdnVolume)) + ", available: ") + toString(swapLimitMax)))
428453 else if (isBlocked)
429454 then emergencyShutdownFAIL()
430455 else {
431456 let leasePart = if ((swapType == "waves"))
432457 then prepareUnleaseAndLease(0)
433458 else nil
434459 $Tuple2(([IntegerEntry(keySwapUserSpentInPeriod(userAddressStr), swapUsdnVolume), IntegerEntry(keyUserLastSwapHeight(userAddressStr), height), IntegerEntry(totalLockedByUserKEY(swapType, userAddressStr), (totalLockedByUser + pmtAmount)), IntegerEntry(getBalanceUnlockBlockKey(userAddressStr), selfUnlockHeight), IntegerEntry(totalLockedKEY(swapType), (totalLocked + pmtAmount)), StringEntry(swapKEY(userAddressStr, txId58), pendingSwapDATA(swapType, pmtAmount, selfUnlockHeight))] ++ leasePart), unit)
435460 }
436461 }
437462
438463
439464 func commonWithdraw (account,index,swapTxId,withdrawTxId) = {
440465 let userAddress = addressFromStringValue(account)
441- let feeManagerAddress = feeManagerAddressREAD()
442466 let dataArray = swapDataFailOrREAD(account, swapTxId)
443467 let selfUnlockHeight = parseIntValue(dataArray[sIdxSelfUnlockHeight])
444468 let swapType = dataArray[sIdxSwapType]
445469 let inAmount = parseIntValue(dataArray[sIdxInAmount])
446470 let swapStatus = dataArray[sIdxStatus]
447471 let startHeight = parseIntValue(dataArray[sIdxStartHeight])
448472 let outFeePart = valueOrElse(getInteger(this, outFeePartKEY(swapType)), DEFAULTSWAPFEE)
449473 let totalLocked = totalLockedREAD(swapType)
450474 let totalLockedByUser = totalLockedByUserREAD(swapType, account)
451475 let unlockHeight = selfUnlockHeight
452476 let indexHeight = getHeightPriceByIndex(index)
453477 let prevIndexHeight = getHeightPriceByIndex((index - 1))
454478 let priceByIndex = getPriceHistory(indexHeight)
455479 let outAmountGrossTuple = if ((swapType == "waves"))
456480 then $Tuple2(convertWavesToNeutrino(inAmount, priceByIndex), neutrinoAssetId)
457481 else if ((swapType == "neutrino"))
458482 then $Tuple2(convertNeutrinoToWaves(inAmount, priceByIndex), unit)
459483 else throw(("Unsupported swap type " + swapType))
460484 let payoutsArray = applyFees(outAmountGrossTuple._1, outFeePart)
461485 let outNetAmount = payoutsArray[IdxNetAmount]
462486 let outFeeAmount = payoutsArray[IdxFeeAmount]
463487 if (isBlocked)
464488 then emergencyShutdownFAIL()
465489 else if ((swapStatus != "PENDING"))
466490 then throw("swap has been already processed")
467491 else if ((unlockHeight > height))
468492 then throw((("please wait for: " + toString(unlockHeight)) + " block height to withdraw funds"))
469493 else if (if (if ((index > priceIndex))
470494 then true
471495 else (unlockHeight > indexHeight))
472496 then true
473497 else if ((prevIndexHeight != 0))
474498 then (prevIndexHeight >= unlockHeight)
475499 else false)
476500 then priceIndexFAIL(index, priceIndex, indexHeight, unlockHeight, prevIndexHeight)
477501 else if ((0 >= payoutsArray[IdxGrossAmount]))
478502 then throw("balance equals zero")
479503 else if (if ((0 > outFeePart))
480504 then true
481505 else (outFeePart >= PAULI))
482506 then throw(((("invalid outFeePart config for " + swapType) + " swap: outFeePart=") + toString(outFeePart)))
483507 else {
484508 let unleaseAmount = if (if ((swapType == "neutrino"))
485509 then (outAmountGrossTuple._1 > 0)
486510 else false)
487511 then outAmountGrossTuple._1
488512 else 0
489513 let state = [IntegerEntry(totalLockedByUserKEY(swapType, account), (totalLockedByUser - inAmount)), IntegerEntry(totalLockedKEY(swapType), (totalLocked - inAmount)), ScriptTransfer(userAddress, outNetAmount, outAmountGrossTuple._2), StringEntry(swapKEY(account, swapTxId), finishSwapDATA(dataArray, priceByIndex, outNetAmount, outFeeAmount, unlockHeight, index, withdrawTxId))]
490514 $Tuple3(state, AttachedPayment(outAmountGrossTuple._2, outFeeAmount), unleaseAmount)
491515 }
492516 }
493517
494518
495519 @Callable(i)
496520 func constructor (neutrinoAssetIdPrm,bondAssetIdPrm,auctionContractPrm,liquidationContractPrm,rpdContractPrm,nodeOracleProviderPubKeyPrm,balanceWavesLockIntervalPrm,balanceNeutrinoLockIntervalPrm,minWavesSwapAmountPrm,minNeutrinoSwapAmountPrm,neutrinoOutFeePartPrm,wavesOutFeePartPrm) = {
497521 let checkCaller = thisOnly(i)
498522 if ((checkCaller == checkCaller))
499523 then if ((size(i.payments) != 0))
500524 then throw("no payments allowed")
501525 else [StringEntry(NeutrinoAssetIdKey, neutrinoAssetIdPrm), StringEntry(BondAssetIdKey, bondAssetIdPrm), StringEntry(AuctionContractKey, auctionContractPrm), StringEntry(LiquidationContractKey, liquidationContractPrm), StringEntry(RPDContractKey, rpdContractPrm), StringEntry(NodeOracleProviderPubKeyKey, nodeOracleProviderPubKeyPrm), IntegerEntry(BalanceWavesLockIntervalKey, balanceWavesLockIntervalPrm), IntegerEntry(BalanceNeutrinoLockIntervalKey, balanceNeutrinoLockIntervalPrm), IntegerEntry(MinWavesSwapAmountKey, minWavesSwapAmountPrm), IntegerEntry(MinNeutrinoSwapAmountKey, minNeutrinoSwapAmountPrm), IntegerEntry(NeutrinoOutFeePartKey, neutrinoOutFeePartPrm), IntegerEntry(WavesOutFeePartKey, wavesOutFeePartPrm)]
502526 else throw("Strict value is not equal to itself.")
503527 }
504528
505529
506530
507531 @Callable(i)
508532 func constructorV2 (mathContract,nsbtStakingContract,swapsTimeframeBlocks) = {
509533 let checkCaller = thisOnly(i)
510534 if ((checkCaller == checkCaller))
511535 then if ((size(i.payments) != 0))
512536 then throw("no payments allowed")
513537 else [StringEntry(MathContractKey, mathContract), StringEntry(NsbtStakingContractKey, nsbtStakingContract), IntegerEntry(swapsTimeframeKEY(), swapsTimeframeBlocks)]
514538 else throw("Strict value is not equal to itself.")
515539 }
516540
517541
518542
519543 @Callable(i)
520544 func swapWavesToNeutrino () = if ((size(i.payments) != 1))
521545 then throw("swapWavesToNeutrino require only one payment")
522546 else {
523547 let pmt = value(i.payments[0])
524548 if (isDefined(pmt.assetId))
525549 then throw("Only Waves token is allowed for swapping.")
526550 else {
527551 let userAddress = toString(i.caller)
528552 let txId58 = toBase58String(i.transactionId)
529553 let swapParamsSTRUCT = asSwapParamsSTRUCT(invoke(this, "swapParamsByUserSYSREADONLY", [userAddress, 0], nil))
530554 let commonSwapResult = commonSwap("waves", pmt.amount, userAddress, txId58, swapParamsSTRUCT)
531555 commonSwapResult
532556 }
533557 }
534558
535559
536560
537561 @Callable(i)
538562 func swapNeutrinoToWaves () = if ((size(i.payments) != 1))
539563 then throw("swapNeutrinoToWaves require only one payment")
540564 else {
541565 let pmt = value(i.payments[0])
542566 if ((pmt.assetId != neutrinoAssetId))
543567 then throw("Only appropriate Neutrino tokens are allowed for swapping.")
544568 else {
545569 let userAddress = toString(i.caller)
546570 let txId58 = toBase58String(i.transactionId)
547571 let swapParamsSTRUCT = asSwapParamsSTRUCT(invoke(this, "swapParamsByUserSYSREADONLY", [userAddress, 0], nil))
548572 let commonSwapResult = commonSwap("neutrino", pmt.amount, userAddress, txId58, swapParamsSTRUCT)
549573 commonSwapResult
550574 }
551575 }
552576
553577
554578
555579 @Callable(i)
556580 func withdraw (account,index,swapTxId) = {
557581 let txId = toBase58String(i.transactionId)
558582 if ((size(i.payments) != 0))
559583 then throw("no payments allowed")
560584 else {
561585 let commonTuple = commonWithdraw(account, index, swapTxId, txId)
562586 let state = commonTuple._1
563587 let fee = commonTuple._2
564588 let unleaseAmt = commonTuple._3
565589 let unleaseInvOrEmpty = invoke(this, "internalUnleaseAndLease", [unleaseAmt], nil)
566590 if ((unleaseInvOrEmpty == unleaseInvOrEmpty))
567- then if ((0 >= fee.amount))
568- then state
569- else {
570- let nsbtStakingReward = invoke(nsbtStakingContract, "deposit", nil, [fee])
571- if ((nsbtStakingReward == nsbtStakingReward))
572- then state
573- else throw("Strict value is not equal to itself.")
574- }
591+ then {
592+ let gnsbtData = asAnyList(invoke(gnsbtControllerContract, "gnsbtInfoSYSREADONLY", [""], nil))
593+ let gnsbtAmtTotal = asInt(gnsbtData[1])
594+ let gnsbtAmtFromSurfTotal = asInt(asAnyList(gnsbtData[3])[3])
595+ let surfFeeAmt = if ((gnsbtAmtTotal != 0))
596+ then fraction(fee.amount, gnsbtAmtFromSurfTotal, gnsbtAmtTotal)
597+ else 0
598+ let nsbtFeeAmt = (fee.amount - surfFeeAmt)
599+ let surfDeposit = if ((surfFeeAmt > 0))
600+ then {
601+ let surfInv = invoke(surfStakingContract, "deposit", nil, [AttachedPayment(fee.assetId, surfFeeAmt)])
602+ if ((surfInv == surfInv))
603+ then nil
604+ else throw("Strict value is not equal to itself.")
605+ }
606+ else nil
607+ if ((surfDeposit == surfDeposit))
608+ then {
609+ let nsbtDeposit = if ((nsbtFeeAmt > 0))
610+ then {
611+ let nsbtInv = invoke(nsbtStakingContract, "deposit", nil, [AttachedPayment(fee.assetId, nsbtFeeAmt)])
612+ if ((nsbtInv == nsbtInv))
613+ then nil
614+ else throw("Strict value is not equal to itself.")
615+ }
616+ else nil
617+ if ((nsbtDeposit == nsbtDeposit))
618+ then state
619+ else throw("Strict value is not equal to itself.")
620+ }
621+ else throw("Strict value is not equal to itself.")
622+ }
575623 else throw("Strict value is not equal to itself.")
576624 }
577625 }
578626
579627
580628
581629 @Callable(i)
582630 func internalUnleaseAndLease (unleaseAmount) = if ((i.caller != this))
583631 then throw("internalUnleaseAndLease is not public method")
584632 else prepareUnleaseAndLease(unleaseAmount)
585633
586634
587635
588636 @Callable(i)
589-func transferUsdnToUser (amount,addr) = if ((i.caller != addressFromStringValue(auctionContract)))
637+func transferUsdnToUser (amount,addr) = if ((i.caller != auctionContract))
590638 then throw("Only auction contract is authorized")
591639 else [ScriptTransfer(addressFromStringValue(addr), amount, neutrinoAssetId)]
592640
593641
594642
595643 @Callable(i)
596-func acceptWaves () = if ((i.caller != addressFromStringValue(auctionContract)))
644+func acceptWaves () = if ((i.caller != auctionContract))
597645 then throw("Currently only auction contract is allowed to call")
598646 else $Tuple2(prepareUnleaseAndLease(0), "success")
599647
600648
601649
602650 @Callable(i)
603651 func approveLeasings (nListS,groupNum) = {
604- let lAmt = (500000 * WAVELET)
652+ let lAmt = (50 * WAVELET)
605653 let nIdxs = [0, 1, 2, 3, 4, 5, 6, 7]
606654 let mngPubS = valueOrElse(getString("%s%s__cfg__leasingManagerPub"), "7AUMX54ukYMYvPmma7yoFf5NjZhs4Bu5nz3Ez9EV8sur")
607655 let mngPub = fromBase58String(mngPubS)
608656 let nodeRegAddrStr = valueOrElse(getString("%s%s__cfg__nodesRegistryAddress"), "3P9vKqQKjUdmpXAfiWau8krREYAY1Xr69pE")
609657 let nodeRegAddr = addressFromStringValue(nodeRegAddrStr)
610658 let lGroupNodeListKEY = getLeaseGroupNodeListKey(groupNum)
611659 let lGrNodeOpt = getString(this, lGroupNodeListKEY)
612660 if (isDefined(lGrNodeOpt))
613661 then throw((("group " + toString(groupNum)) + " already initialized"))
614662 else {
615663 let nList = split(nListS, SEP)
616664 let expCount = size(nIdxs)
617665 if ((i.callerPublicKey != mngPub))
618666 then throw("approveLeasings not authorized")
619667 else {
620- let $t02735227414 = readNodeInfo(0)
621- let nAddr0 = $t02735227414._1
622- let lAmtKEY0 = $t02735227414._2
623- let lAmt0 = $t02735227414._3
624- let lIdKEY0 = $t02735227414._4
625- let lId0 = $t02735227414._5
626- let $t02741727479 = readNodeInfo(1)
627- let nAddr1 = $t02741727479._1
628- let lAmtKEY1 = $t02741727479._2
629- let lAmt1 = $t02741727479._3
630- let lIdKEY1 = $t02741727479._4
631- let lId1 = $t02741727479._5
668+ let $t02837828440 = readNodeInfo(0)
669+ let nAddr0 = $t02837828440._1
670+ let lAmtKEY0 = $t02837828440._2
671+ let lAmt0 = $t02837828440._3
672+ let lIdKEY0 = $t02837828440._4
673+ let lId0 = $t02837828440._5
674+ let $t02844328505 = readNodeInfo(1)
675+ let nAddr1 = $t02844328505._1
676+ let lAmtKEY1 = $t02844328505._2
677+ let lAmt1 = $t02844328505._3
678+ let lIdKEY1 = $t02844328505._4
679+ let lId1 = $t02844328505._5
632680 let newL0 = Lease(nAddr0, (lAmt0 - ((lAmt * expCount) / 2)))
633681 let newL1 = Lease(nAddr1, (lAmt1 - ((lAmt * expCount) / 2)))
634682 let validation = invoke(nodeRegAddr, "validateAndApproveLeasings", [nListS], nil)
635683 if ((validation == validation))
636684 then {
637685 func forEachNodeValidateAndGenerateLease (a,i) = {
638686 let node = nList[i]
639687 let la = Lease(addressFromStringValue(node), lAmt)
640688 (a ++ [la, BinaryEntry(getLeaseIdByAddressKey(node), lcalc(la)), IntegerEntry(getLeaseAmountByAddressKey(node), lAmt)])
641689 }
642690
643691 ([StringEntry(lGroupNodeListKEY, nListS), BinaryEntry(lIdKEY0, lcalc(newL0)), BinaryEntry(lIdKEY1, lcalc(newL1)), IntegerEntry(lAmtKEY0, newL0.amount), IntegerEntry(lAmtKEY1, newL1.amount), LeaseCancel(lId0), LeaseCancel(lId1), newL0, newL1] ++ {
644692 let $l = nIdxs
645693 let $s = size($l)
646694 let $acc0 = nil
647695 func $f0_1 ($a,$i) = if (($i >= $s))
648696 then $a
649697 else forEachNodeValidateAndGenerateLease($a, $l[$i])
650698
651699 func $f0_2 ($a,$i) = if (($i >= $s))
652700 then $a
653701 else throw("List size exceeds 8")
654702
655703 $f0_2($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($acc0, 0), 1), 2), 3), 4), 5), 6), 7), 8)
656704 })
657705 }
658706 else throw("Strict value is not equal to itself.")
659707 }
660708 }
661709 }
662710
663711
664712
665713 @Callable(i)
666-func swapParamsByUserSYSREADONLY (userAddressStr,nsbtDiff) = {
667- let nsbtData = asAnyList(invoke(nsbtStakingContract, "nsbtStakingSYSREADONLY", [userAddressStr], nil))
668- if ((nsbtData == nsbtData))
669- then {
670- let gnsbtAmt = (asInt(nsbtData[0]) + nsbtDiff)
671- let gnsbtAmtTotal = (asInt(nsbtData[1]) + nsbtDiff)
672- let swapLimitMax = asInt(invoke(mathContract, "calcSwapLimitREADONLY", [gnsbtAmt], nil))
673- let lastSwapHeight = valueOrElse(getInteger(this, keyUserLastSwapHeight(userAddressStr)), 0)
674- let swapLimitTimelifeBlocks = swapsTimeframeREAD()
675- let passedBlocksAfterLastSwap = (height - lastSwapHeight)
676- let isSwapTimelifeNew = (passedBlocksAfterLastSwap >= swapLimitTimelifeBlocks)
677- let swapLimitSpent = if (isSwapTimelifeNew)
678- then 0
679- else valueOrElse(getInteger(this, keySwapUserSpentInPeriod(userAddressStr)), 0)
680- let blcks2LmtReset = if (isSwapTimelifeNew)
681- then 0
682- else (swapLimitTimelifeBlocks - passedBlocksAfterLastSwap)
683- $Tuple2(nil, $Tuple5(swapLimitMax, swapLimitSpent, blcks2LmtReset, gnsbtAmt, gnsbtAmtTotal))
684- }
685- else throw("Strict value is not equal to itself.")
714+func swapParamsByUserSYSREADONLY (userAddressStr,gnsbtDiff) = {
715+ let gnsbtData = asAnyList(invoke(gnsbtControllerContract, "gnsbtInfoSYSREADONLY", [userAddressStr], nil))
716+ let gnsbtAmt = (asInt(gnsbtData[0]) + gnsbtDiff)
717+ let gnsbtAmtTotal = (asInt(gnsbtData[1]) + gnsbtDiff)
718+ let swapLimitMax = asInt(invoke(mathContract, "calcSwapLimitREADONLY", [gnsbtAmt], nil))
719+ let lastSwapHeight = valueOrElse(getInteger(this, keyUserLastSwapHeight(userAddressStr)), 0)
720+ let swapLimitTimelifeBlocks = swapsTimeframeREAD()
721+ let passedBlocksAfterLastSwap = (height - lastSwapHeight)
722+ let isSwapTimelifeNew = (passedBlocksAfterLastSwap >= swapLimitTimelifeBlocks)
723+ let swapLimitSpent = if (isSwapTimelifeNew)
724+ then 0
725+ else valueOrElse(getInteger(this, keySwapUserSpentInPeriod(userAddressStr)), 0)
726+ let blcks2LmtReset = if (isSwapTimelifeNew)
727+ then 0
728+ else (swapLimitTimelifeBlocks - passedBlocksAfterLastSwap)
729+ $Tuple2(nil, $Tuple5(swapLimitMax, swapLimitSpent, blcks2LmtReset, gnsbtAmt, gnsbtAmtTotal))
730+ }
731+
732+
733+
734+@Callable(i)
735+func updateReservesAndNeutrinoSupply () = {
736+ func getNumberByKeyInternal (key) = match getInteger(this, key) {
737+ case a: Int =>
738+ a
739+ case _ =>
740+ 0
741+ }
742+
743+ let nMetrix = asAnyList(invoke(mathContract, "calcNeutinoMetricsREADONLY", nil, nil))
744+ let idx = getNumberByKeyInternal("updateReservesAndNeutrinoSupplyIdx")
745+ let newIdx = (idx + 1)
746+[IntegerEntry("updateReservesAndNeutrinoSupplyIdx", newIdx), IntegerEntry("reserve", asInt(nMetrix[3])), IntegerEntry("neutrinoSupply", asInt(nMetrix[5])), IntegerEntry("reservesInUsdn", convertWavesToNeutrino(asInt(nMetrix[3]), asInt(nMetrix[0]))), IntegerEntry("surplus", asInt(nMetrix[6])), IntegerEntry("deficit", -(asInt(nMetrix[6])))]
747+ }
748+
749+
750+
751+@Callable(i)
752+func wavesBalancesVsPayment () = {
753+ let b = wavesBalance(this)
754+[IntegerEntry("wavesBalance_available", b.available), IntegerEntry("wavesBalance_regular", b.regular), IntegerEntry("wavesBalance_generating", b.generating), IntegerEntry("wavesBalance_effective", b.effective), IntegerEntry("waves_payment", value(i.payments[0]).amount)]
686755 }
687756
688757
689758 @Verifier(tx)
690759 func verify () = {
691760 let id = toBase58String(tx.id)
761+ let pubKeyAdminsListStr = makeString(["ExtEEK19nmKj9mCpnWyvEEJFYATLMcVEMvohhUHkyHNm", "Ev5py5FfBQX9cZpYKnfQrTB49Byf8QmpZWeDVRim4yV7", "DUuuLjXu98nBwZc7fqwCTjtA3nnRwgTbkMSr5SU2NmDR", "DUuuLjXu98nBwZc7fqwCTjtA3nnRwgTbkMSr5SU2NmDR"], SEP)
762+ let pubKeyAdminsList = split(valueOrElse(getString(controlContract, "%s__multisig"), pubKeyAdminsListStr), SEP)
692763 let count = ((((if (sigVerify(tx.bodyBytes, tx.proofs[0], fromBase58String(pubKeyAdminsList[0])))
693764 then 1
694765 else 0) + (if (sigVerify(tx.bodyBytes, tx.proofs[1], fromBase58String(pubKeyAdminsList[1])))
695766 then 1
696767 else 0)) + (if (sigVerify(tx.bodyBytes, tx.proofs[2], fromBase58String(pubKeyAdminsList[2])))
697768 then 1
698769 else 0)) + (if (sigVerify(tx.bodyBytes, tx.proofs[3], fromBase58String(pubKeyAdminsList[3])))
699770 then 2
700771 else 0))
701772 match tx {
702773 case sponsorTx: SponsorFeeTransaction =>
703774 if (checkIsValidMinSponsoredFee(sponsorTx))
704775 then (count >= 3)
705776 else false
706777 case _ =>
707778 (count >= 3)
708779 }
709780 }
710781

github/deemru/w8io/3ef1775 
284.65 ms