Compare commits

...

52 Commits

Author SHA1 Message Date
poka 4dc34fb64e update requirements 2023-09-15 10:16:33 +02:00
poka cb7037072e add dict annotation 2023-09-13 09:06:44 +02:00
poka 1737080b82 fix command with no args 2023-09-10 11:33:31 +02:00
poka 56f5972ad0 refacto args passing 2023-09-10 09:44:52 +02:00
poka 5fe53c1d4c refacto arguments parsing 2023-09-10 08:15:56 +02:00
poka 919a4171f5 bump 0.1.0 2023-09-10 02:23:44 +02:00
poka 6dc86592b9 refacto and translate main file 2023-09-10 02:03:42 +02:00
poka 595f7722bf refacto #1 2023-09-10 01:53:38 +02:00
poka e188d4da94 unindent json 2023-09-09 22:35:35 +02:00
poka aa6fd2b738 antispam: improve condition 2023-09-09 13:26:27 +02:00
poka 676fbbfa9b antispam: remove unknown wallet by GVA 2023-09-09 12:51:47 +02:00
poka 3492643d4f add map option to walletList 2023-09-09 10:14:31 +02:00
poka b1049bd7b7 Merge branch 'refactoGeolocProfiles' 2023-09-09 09:50:52 +02:00
poka 2c5435a18b add sublevel in json, with timestamp 2023-09-09 09:50:20 +02:00
poka 51ae54484c remove unused imports 2023-09-09 09:44:27 +02:00
poka e0ba392671 add geolocProfiles command 2023-09-09 09:38:22 +02:00
poka ab50340595 update vscode settings 2023-09-09 04:26:50 +02:00
poka b89456a9b4 Merge branch 'g1walletbot' 2023-09-09 00:16:01 +02:00
Kapis a13723fa3c showing hash in json output 2023-03-16 18:45:34 +01:00
poka bdc8b938fc Merge pull request 'Adding blockstamp to history JSON' (#24) from g1walletbot into master
Reviewed-on: axiom-team/jaklis#24
2023-03-14 21:20:18 +01:00
Kapis fc587819b0 Adding blockstamp to history JSON 2023-03-14 20:35:17 +01:00
poka b6dbc5f8da Merge pull request 'JSON history added field status and new GVA node' (#23) from kapis/jaklis:master into master
Reviewed-on: axiom-team/jaklis#23
2023-03-11 21:18:20 +01:00
Kapis dbbd2c3ed6 more info about dunikey path 2023-03-11 21:12:57 +01:00
Kapis 31f4fc1f83 added Poka node 2023-03-10 22:21:33 +01:00
Kapis 4a1cae165a Added pini GVA node 2023-03-10 22:21:33 +01:00
Kapis 249aa2c6a7 adding status to json history output 2023-03-10 22:21:33 +01:00
Kapis afd244d405 ignoring pycache 2023-03-10 22:21:33 +01:00
poka b047142b64 change default duniter node 2023-03-10 22:20:58 +01:00
poka 0c264fc57c hello aya 2023-01-21 11:31:19 +01:00
poka afad8092b7 fix setup symlink 2023-01-21 11:29:29 +01:00
poka f99b6558a2 fix fucking down CS POD in .env.template 2022-11-23 01:41:04 +01:00
poka a882593ead update .env.template 2022-10-23 04:27:00 +02:00
poka 8690615335 fix GVA comment transaction sentence 2022-10-23 04:21:30 +02:00
poka 60bae39fba fix everything in universe and more 2022-10-23 04:05:48 +02:00
poka 0d7db5ba17 remove tata mother fucker 2022-10-23 03:44:37 +02:00
poka 8daee665e4 fix bad nonce on some message 2022-10-23 03:42:18 +02:00
poka 5a7b0c6901 fix null message content 2022-10-23 03:31:27 +02:00
poka ec7fcdfb45 Replace NODE and POD envvar with DUNITER and ESNODE, like g1stats 2022-03-29 00:27:00 +02:00
poka 9117def4b9 Change cesium pods and duniter node ofr good one; Fix MY_PATH for symlink; Add jaklis to env in setup.sh 2021-12-08 11:12:51 +01:00
poka efed7354df listWallets: Show balance in G1, not cents 2021-10-18 01:31:28 +02:00
poka db39e5c77d Mise à jour de 'README.md' 2021-10-09 12:51:26 +02:00
poka b291ddaef5 test git 2021-10-07 10:42:01 +02:00
poka 9270fb7ba2 Add username for non mbr 2021-08-28 06:56:29 +02:00
poka 53cbdfe982 Add username in mbr brut list 2021-08-28 03:11:11 +02:00
poka e8aa6723ee Improve listWallets options and algo 2021-08-28 02:47:20 +02:00
poka 314dc74404 fix cursor 2021-08-27 23:08:38 +02:00
poka 7bfa733fc5 fix typo 2021-08-27 23:02:08 +02:00
poka 43e5836eef fix listWallets with no dunikey 2021-08-27 22:59:16 +02:00
poka 116b362ea9 Add listWallet command 2021-08-27 05:02:16 +02:00
poka a6f26331d7 Merge pull request 'default offer category = CD/DVD' (#17) from qo-op/jaklis:master into master
Reviewed-on: axiom-team/jaklis#17
2021-07-02 01:14:20 +02:00
Jean-Jacques Brucker b7fa622941 finish previous shebang fix + minor improvement in code and help message 2021-07-01 11:24:40 +02:00
poka fecc636418 Merge pull request 'To work with python 3.6 on Linux Mint' (#16) from qo-op/jaklis:master into master
Reviewed-on: axiom-team/jaklis#16
2021-07-01 02:24:40 +02:00
18 changed files with 971 additions and 296 deletions

View File

@ -1,10 +1,16 @@
# Chemin de la clé privé Ḡ1 de l'émetteur, au format PubSec # Chemin de la clé privé Ḡ1 de l'émetteur, au format PubSec
DUNIKEY= # ex. DUNIKEY="/path/myprivateG1key.dunikey"
DUNIKEY=""
# Noeud Duniter # Noeud Duniter
NODE=https://g1.librelois.fr/gva DUNITER="https://g1.asycn.io"
#DUNITER="https://duniter.pini.fr"
#DUNITER="https://g1v1.p2p.legal"
# Adresse du pod Cesium ou Gchange à utiliser # Adresse du pod Cesium ou Gchange à utiliser
POD=https://g1.data.le-sou.org ESNODE="https://g1.data.e-is.pro"
#POD=https://g1.data.duniter.fr #ESNODE="https://g1.data.presles.fr"
#POD=https://data.gchange.fr #ESNODE="https://g1.data.adn.life"
#ESNODE=https://g1.data.le-sou.org
#ESNODE=https://g1.data.duniter.fr
#ESNODE=https://data.gchange.fr

4
.gitignore vendored
View File

@ -1,5 +1,5 @@
.env .env
__pycache__
*.dunikey *.dunikey
not4U not4U
.vscode/settings.json
__pycache__

6
.vscode/settings.json vendored Normal file → Executable file
View File

@ -1,3 +1,7 @@
{ {
"python.pythonPath": "/usr/bin/python3.9" "python.pythonPath": "/usr/bin/python3.9",
"[python]": {
"editor.defaultFormatter": "ms-python.black-formatter"
},
"python.formatting.provider": "none"
} }

View File

@ -13,7 +13,7 @@ Débrouillez-vous.
## Utilisation ## Utilisation
*Python 3.9 minimum* *Python 3.6 minimum*
Renseignez optionnellement le fichier **.env** (Généré lors de la première tentative d'execution, ou à copier depuis .env.template). Renseignez optionnellement le fichier **.env** (Généré lors de la première tentative d'execution, ou à copier depuis .env.template).

546
jaklis.py
View File

@ -1,191 +1,395 @@
#!/usr/bin/env python3 #!/usr/bin/env python3
import argparse, sys, os, getpass, string, random import argparse
from os.path import join, dirname import sys
from shutil import copyfile import os
import string
import random
from dotenv import load_dotenv from dotenv import load_dotenv
from duniterpy.key import SigningKey from duniterpy.key import SigningKey
from pathlib import Path
from lib.gva import GvaApi
from lib.cesium import CesiumPlus
__version__ = "0.0.4" __version__ = "0.1.1"
MY_PATH = os.path.realpath(os.path.dirname(sys.argv[0])) + '/' MY_PATH = Path(__file__).resolve().parent
# Get variables environment # Set file paths
if not os.path.isfile(MY_PATH + '.env'): dotenv_file = MY_PATH / ".env"
copyfile(MY_PATH + ".env.template",MY_PATH + ".env") dotenv_template = MY_PATH / ".env.template"
dotenv_path = join(dirname(__file__),MY_PATH + '.env')
load_dotenv(dotenv_path)
# Parse arguments # Check and create dotenv file
parser = argparse.ArgumentParser(description="Client CLI pour Cesium+ et Ḡchange") if not dotenv_file.is_file():
parser.add_argument('-v', '--version', action='store_true', help="Affiche la version actuelle du programme") dotenv_file.write_text(dotenv_template.read_text())
parser.add_argument('-k', '--key', help="Chemin vers mon trousseau de clé (PubSec)")
parser.add_argument('-n', '--node', help="Adresse du noeud Cesium+, Gchange ou Duniter à utiliser")
subparsers = parser.add_subparsers(title="Commandes de jaklis", dest="cmd") # Load environment variables
read_cmd = subparsers.add_parser('read', help="Lecture des messages") load_dotenv(dotenv_file)
send_cmd = subparsers.add_parser('send', help="Envoi d'un message")
delete_cmd = subparsers.add_parser('delete', help="Supression d'un message")
getProfile_cmd = subparsers.add_parser('get', help="Voir un profile Cesium+")
setProfile_cmd = subparsers.add_parser('set', help="Configurer son profile Cesium+")
eraseProfile_cmd = subparsers.add_parser('erase', help="Effacer son profile Cesium+")
stars_cmd = subparsers.add_parser('stars', help="Voir les étoiles d'un profile / Noter un profile (option -s NOTE)")
unstars_cmd = subparsers.add_parser('unstars', help="Supprimer un star")
getoffer_cmd = subparsers.add_parser('getoffer', help="Obtenir les informations d'une annonce gchange")
setoffer_cmd = subparsers.add_parser('setoffer', help="Créer une annonce gchange")
deleteoffer_cmd = subparsers.add_parser('deleteoffer', help="Supprimer une annonce gchange")
pay_cmd = subparsers.add_parser('pay', help="Payer en Ḡ1")
history_cmd = subparsers.add_parser('history', help="Voir l'historique des transactions d'un compte Ḡ1")
balance_cmd = subparsers.add_parser('balance', help="Voir le solde d'un compte Ḡ1")
id_cmd = subparsers.add_parser('id', help="Voir l'identité d'une clé publique/username")
id_balance_cmd = subparsers.add_parser('idBalance', help="Voir l'identité d'une clé publique/username et son solde")
currentUd = subparsers.add_parser('currentUd', help="Affiche la montant actuel du dividende Universel")
# Messages management # Set global values (default parameters) regarding environment variables
read_cmd.add_argument('-n', '--number',type=int, default=3, help="Affiche les NUMBER derniers messages") node = os.getenv("DUNITER") + "/gva" or "https://g1v1.p2p.legal/gva"
read_cmd.add_argument('-j', '--json', action='store_true', help="Sort au format JSON") pod = os.getenv("ESNODE") or "https://g1.data.e-is.pro"
read_cmd.add_argument('-o', '--outbox', action='store_true', help="Lit les messages envoyés") destPubkey = False
send_cmd.add_argument('-d', '--destinataire', required=True, help="Destinataire du message") # define parser
send_cmd.add_argument('-t', '--titre', help="Titre du message à envoyer") parser = argparse.ArgumentParser(
send_cmd.add_argument('-m', '--message', help="Message à envoyer") description="CLI Client for Cesium+ and Ḡchange",
send_cmd.add_argument('-f', '--fichier', help="Envoyer le message contenu dans le fichier 'FICHIER'") epilog="current node: '" + node + "', current pod: '" + pod + "'.",
send_cmd.add_argument('-o', '--outbox', action='store_true', help="Envoi le message sur la boite d'envoi") )
delete_cmd.add_argument('-i', '--id', action='append', nargs='+', required=True, help="ID(s) du/des message(s) à supprimer") # load global arguments
delete_cmd.add_argument('-o', '--outbox', action='store_true', help="Suppression d'un message envoyé") parser.add_argument(
"-v",
"--version",
action="store_true",
help="Display the current program version",
)
parser.add_argument("-k", "--key", help="Path to the keyfile (PubSec)")
parser.add_argument(
"-n", "--node", help="Address of the Cesium+, Gchange, or Duniter node to use"
)
# Profiles management # Define commands with arguments
setProfile_cmd.add_argument('-n', '--name', help="Nom du profile") commands = {
setProfile_cmd.add_argument('-d', '--description', help="Description du profile") "read": {
setProfile_cmd.add_argument('-v', '--ville', help="Ville du profile") "help": "Read messages",
setProfile_cmd.add_argument('-a', '--adresse', help="Adresse du profile") "arguments": {
setProfile_cmd.add_argument('-pos', '--position', nargs=2, help="Points géographiques (lat + lon)") ("n", "number"): {
setProfile_cmd.add_argument('-s', '--site', help="Site web du profile") "type": int,
setProfile_cmd.add_argument('-A', '--avatar', help="Chemin vers mon avatar en PNG") "default": 3,
"help": "Display the last NUMBER messages",
getProfile_cmd.add_argument('-p', '--profile', help="Nom du profile") },
getProfile_cmd.add_argument('-a', '--avatar', action='store_true', help="Récupérer également l'avatar au format raw base64") ("j", "json"): {"action": "store_true", "help": "Output in JSON format"},
("o", "outbox"): {"action": "store_true", "help": "Read sent messages"},
# Likes management },
stars_cmd.add_argument('-p', '--profile', help="Profile cible") "type": "cesium",
stars_cmd.add_argument('-n', '--number', type=int, help="Nombre d'étoile") },
unstars_cmd.add_argument('-p', '--profile', help="Profile à dénoter") "send": {
"help": "Send a message",
# Offers management "arguments": {
getoffer_cmd.add_argument('-i', '--id', help="Annonce cible à récupérer") ("d", "destinataire"): {
setoffer_cmd.add_argument('-t', '--title', help="Titre de l'annonce à créer") "required": True,
setoffer_cmd.add_argument('-d', '--description', help="Description de l'annonce à créer") "help": "Recipient of the message",
setoffer_cmd.add_argument('-c', '--category', help="Categorie de l'annonce à créer") },
setoffer_cmd.add_argument('-l', '--localisation', nargs=2, help="Localisation de l'annonce à créer (lat + lon)") ("t", "titre"): {"help": "Title of the message to send"},
setoffer_cmd.add_argument('-p', '--picture', help="Image de l'annonce à créer") ("m", "message"): {"help": "Message to send"},
setoffer_cmd.add_argument('-ci', '--city', help="Ville de l'annonce à créer") ("f", "fichier"): {"help": "Send the message from the 'FILE'"},
setoffer_cmd.add_argument('-pr', '--price', help="Prix de l'annonce à créer") ("o", "outbox"): {
deleteoffer_cmd.add_argument('-i', '--id', help="Annonce cible à supprimer") "action": "store_true",
"help": "Send the message to the outbox",
# GVA usage },
pay_cmd.add_argument('-p', '--pubkey', help="Destinataire du paiement") },
pay_cmd.add_argument('-a', '--amount', type=float, help="Montant de la transaction") "type": "cesium",
pay_cmd.add_argument('-c', '--comment', default="", help="Commentaire de la transaction") },
pay_cmd.add_argument('-m', '--mempool', action='store_true', help="Utilise les sources en Mempool") "delete": {
pay_cmd.add_argument('-v', '--verbose', action='store_true', help="Affiche le résultat JSON de la transaction") "help": "Delete a message",
"arguments": {
history_cmd.add_argument('-p', '--pubkey', help="Clé publique du compte visé") ("i", "id"): {
history_cmd.add_argument('-n', '--number',type=int, default=10, help="Affiche les NUMBER dernières transactions") "action": "append",
history_cmd.add_argument('-j', '--json', action='store_true', help="Affiche le résultat en format JSON") "nargs": "+",
history_cmd.add_argument('--nocolors', action='store_true', help="Affiche le résultat en noir et blanc") "required": True,
"help": "ID(s) of the message(s) to delete",
balance_cmd.add_argument('-p', '--pubkey', help="Clé publique du compte visé") },
balance_cmd.add_argument('-m', '--mempool', action='store_true', help="Utilise les sources en Mempool") ("o", "outbox"): {
id_cmd.add_argument('-p', '--pubkey', help="Clé publique du compte visé") "action": "store_true",
id_cmd.add_argument('-u', '--username', help="Username du compte visé") "help": "Delete a sent message",
id_balance_cmd.add_argument('-p', '--pubkey', help="Pubkey du compte visé") },
currentUd.add_argument('-p', '--pubkey', help="Pubkey du compte visé") },
"type": "cesium",
},
"get": {
"help": "View a Cesium+ profile",
"arguments": {
("p", "profile"): {"help": "Profile name"},
("a", "avatar"): {
"action": "store_true",
"help": "Also retrieve the avatar in raw base64 format",
},
},
"type": "cesium",
},
"page": {
"help": "View a Cesium+ page",
"arguments": {
("p", "page"): {"help": "Page name"},
("a", "avatar"): {
"action": "store_true",
"help": "Also retrieve the page's avatar in raw base64 format",
},
},
"type": "cesium",
},
"set": {
"help": "Configure your Cesium+ profile",
"arguments": {
("n", "name"): {"help": "Profile name"},
("d", "description"): {"help": "Profile description"},
("v", "ville"): {"help": "Profile city"},
("a", "adresse"): {"help": "Profile address"},
("pos", "position"): {
"nargs": 2,
"help": "Geographical coordinates (lat + lon)",
},
("s", "site"): {"help": "Profile website"},
("A", "avatar"): {"help": "Path to profile avatar in PNG"},
},
"type": "cesium",
},
"erase": {
"help": "Erase your Cesium+ profile",
"arguments": {},
"type": "cesium",
},
"stars": {
"help": "View a profile's stars / Rate a profile (option -s RATING)",
"arguments": {
("p", "profile"): {"help": "Target profile"},
("n", "number"): {"type": int, "help": "Number of stars"},
},
"type": "cesium",
},
"unstars": {
"help": "Remove a star",
"arguments": {
("p", "profile"): {"help": "Profile to unstar"},
},
"type": "cesium",
},
"getoffer": {
"help": "Get information about a Ḡchange listing",
"arguments": {
("i", "id"): {"help": "Target listing to retrieve"},
},
"type": "cesium",
},
"setoffer": {
"help": "Create a Ḡchange listing",
"arguments": {
("t", "title"): {"help": "Title of the listing to create"},
("d", "description"): {"help": "Description of the listing to create"},
("c", "category"): {"help": "Category of the listing to create"},
("l", "location"): {
"nargs": 2,
"help": "Location of the listing to create (lat + lon)",
},
("p", "picture"): {"help": "Image of the listing to create"},
("ci", "city"): {"help": "City of the listing to create"},
("pr", "price"): {"help": "Price of the listing to create"},
},
"type": "cesium",
},
"deleteoffer": {
"help": "Delete a Ḡchange listing",
"arguments": {
("i", "id"): {"help": "Target listing to delete"},
},
"type": "cesium",
},
"geolocProfiles": {
"help": "Get JSON of all geolocated accounts",
"arguments": {},
"type": "cesium",
},
"pay": {
"help": "Pay in Ḡ1",
"arguments": {
("p", "pubkey"): {"help": "Payment recipient"},
("a", "amount"): {"type": float, "help": "Transaction amount"},
("c", "comment"): {
"default": "",
"help": "Transaction comment",
"nargs": "*",
},
("m", "mempool"): {
"action": "store_true",
"help": "Use mempool sources",
},
("v", "verbose"): {
"action": "store_true",
"help": "Display the JSON result of the transaction",
},
},
"type": "gva",
},
"history": {
"help": "View Ḡ1 account transaction history",
"arguments": {
("p", "pubkey"): {"help": "Public key of the target account"},
("n", "number"): {
"type": int,
"default": 10,
"help": "Display the last NUMBER transactions",
},
("j", "json"): {
"action": "store_true",
"help": "Display the result in JSON format",
},
("nocolors"): {
"action": "store_true",
"help": "Display the result in black and white",
},
},
"type": "gva",
},
"balance": {
"help": "View Ḡ1 account balance",
"arguments": {
("p", "pubkey"): {"help": "Public key of the target account"},
("m", "mempool"): {
"action": "store_true",
"help": "Use mempool sources",
},
},
"type": "gva",
},
"id": {
"help": "View public key/username identity",
"arguments": {
("p", "pubkey"): {"help": "Public key of the target account"},
("u", "username"): {"help": "Username of the target account"},
},
"type": "gva",
},
"idBalance": {
"help": "View public key/username identity and balance",
"arguments": {
("p", "pubkey"): {"help": "Public key of the target account"},
},
"type": "gva",
},
"currentUd": {
"help": "Display the current Universal Dividend amount",
"arguments": {
("p", "pubkey"): {"help": "Public key of the target account"},
},
"type": "gva",
},
"listWallets": {
"help": "List all G1 wallets",
"arguments": {
("m", "mbr"): {
"action": "store_true",
"help": "Display raw list of member pubkeys",
},
("nm", "non_mbr"): {
"action": "store_true",
"help": "Display raw list of nonmember identity pubkeys",
},
("l", "larf"): {
"action": "store_true",
"help": "Display raw list of nonmember pubkeys",
},
("b", "brut"): {
"action": "store_true",
"help": "Display raw list of all pubkeys",
},
},
"type": "gva",
},
}
# Process commands and arguments
subparsers = parser.add_subparsers(title="jaklis Commands", dest="cmd")
for cmd, cmd_info in commands.items():
cmd_parser = subparsers.add_parser(cmd, help=cmd_info["help"])
for args, kwargs in cmd_info["arguments"].items():
if isinstance(args, str):
cmd_parser.add_argument("--" + args, **kwargs)
else:
short_arg, long_arg = args
cmd_parser.add_argument("-" + short_arg, "--" + long_arg, **kwargs)
args = parser.parse_args() args = parser.parse_args()
args_dict = vars(args)
cmd = args.cmd cmd = args.cmd
if args.version: if args.version:
print(__version__) print(__version__)
sys.exit(0) sys.exit(0)
if not cmd: if not cmd:
parser.print_help() parser.print_help()
sys.exit(1) sys.exit(0)
def createTmpDunikey(): def createTmpDunikey():
# Generate pseudo-random nonce # Generate a pseudo-random nonce
nonce=[] nonce = "".join(
for _ in range(32): random.choice(string.ascii_letters + string.digits) for _ in range(32)
nonce.append(random.choice(string.ascii_letters + string.digits)) )
nonce = ''.join(nonce)
keyPath = "/tmp/secret.dunikey-" + nonce keyPath = "/tmp/secret.dunikey-" + nonce
key = SigningKey.from_credentials(getpass.getpass("Identifiant: "), getpass.getpass("Mot de passe: "), None) # Create a dummy key (replace with actual key creation logic)
key = SigningKey.from_credentials(
"sgse547yhd54xv6541srdh", "sfdgwdrhpkxdawsbszqpof1sdg65xc", None
)
key.save_pubsec_file(keyPath) key.save_pubsec_file(keyPath)
return keyPath return keyPath
# Check if we need dunikey
try:
pubkey = args.pubkey
except:
pubkey = False
try:
profile = args.profile
except:
profile = False
if cmd in ('history','balance','get','id','idBalance') and (pubkey or profile): def get_arg_value(args, arg):
noNeedDunikey = True
keyPath = False
try: try:
dunikey = args.pubkey return getattr(args, arg)
except: except AttributeError:
dunikey = args.profile return False
else:
noNeedDunikey = False
def get_dunikey(args):
if args.key: if args.key:
dunikey = args.key return args.key
keyPath = False dunikey = os.getenv("DUNIKEY")
else:
dunikey = os.getenv('DUNIKEY')
if not dunikey: if not dunikey:
keyPath = createTmpDunikey() keyPath = createTmpDunikey()
dunikey = keyPath dunikey = keyPath
else:
keyPath = False
if not os.path.isfile(dunikey): if not os.path.isfile(dunikey):
HOME = os.getenv("HOME") HOME = os.getenv("HOME")
dunikey = HOME + dunikey dunikey = HOME + dunikey
if not os.path.isfile(dunikey): if not os.path.isfile(dunikey):
sys.stderr.write('Le fichier de trousseau {0} est introuvable.\n'.format(dunikey)) sys.stderr.write("The keyfile {0} is not found.\n".format(dunikey))
sys.exit(1) sys.exit(1)
return dunikey
# Construct CesiumPlus object pubkey = get_arg_value(args, "pubkey")
if cmd in ("read","send","delete","set","get","erase","stars","unstars","getoffer","setoffer","deleteoffer"): profile = get_arg_value(args, "profile")
from lib.cesium import CesiumPlus
if args.node: noNeedDunikey = cmd in (
pod = args.node "history",
else: "balance",
pod = os.getenv('POD') "page",
if not pod: "id",
pod="https://g1.data.le-sou.org" "idBalance",
"listWallets",
"geolocProfiles",
) and (pubkey or profile)
cesium = CesiumPlus(dunikey, pod, noNeedDunikey) if noNeedDunikey:
dunikey = pubkey if pubkey else profile
else:
dunikey = get_dunikey(args)
keyPath = False if dunikey else createTmpDunikey()
def handle_cesium_commands(args, cmd, cesium):
# Get args of the command
cmd_args = (
list(zip(*list(commands[cmd]["arguments"].keys())))[1]
if len(commands[cmd]["arguments"].keys()) > 0
else []
)
cmd_args_dict = {arg: args_dict[arg] for arg in cmd_args if arg in args_dict}
cmd_args_values = list(cmd_args_dict.values())
# Messaging # Messaging
if cmd == "read": if cmd == "read":
cesium.read(args.number, args.outbox, args.json) cesium.read(*cmd_args_values)
elif cmd == "send": elif cmd == "send":
if args.fichier: if args.fichier:
with open(args.fichier, 'r') as f: with open(args.fichier, "r") as f:
msgT = f.read() msgT = f.read()
titre = msgT.splitlines(True)[0].replace('\n', '') titre = msgT.splitlines(True)[0].replace("\n", "")
msg = ''.join(msgT.splitlines(True)[1:]) msg = "".join(msgT.splitlines(True)[1:])
if args.titre: if args.titre:
titre = args.titre titre = args.titre
msg = msgT msg = msgT
@ -193,8 +397,8 @@ if cmd in ("read","send","delete","set","get","erase","stars","unstars","getoffe
titre = args.titre titre = args.titre
msg = args.message msg = args.message
else: else:
titre = input("Indiquez le titre du message: ") titre = input("Enter the message title: ")
msg = input("Indiquez le contenu du message: ") msg = input("Enter the message content: ")
cesium.send(titre, msg, args.destinataire, args.outbox) cesium.send(titre, msg, args.destinataire, args.outbox)
@ -203,11 +407,15 @@ if cmd in ("read","send","delete","set","get","erase","stars","unstars","getoffe
# Profiles # Profiles
elif cmd == "set": elif cmd == "set":
cesium.set(args.name, args.description, args.ville, args.adresse, args.position, args.site, args.avatar) cesium.set(**cmd_args_dict)
elif cmd == "get": elif cmd == "get":
cesium.get(args.profile, args.avatar) cesium.get(**cmd_args_dict)
elif cmd == "page":
cesium.getPage(**cmd_args_dict)
elif cmd == "erase": elif cmd == "erase":
cesium.erase() cesium.erase()
elif cmd == "geolocProfiles":
cesium.geolocProfiles(node)
# Stars # Stars
elif cmd == "stars": elif cmd == "stars":
@ -222,27 +430,22 @@ if cmd in ("read","send","delete","set","get","erase","stars","unstars","getoffe
elif cmd == "getoffer": elif cmd == "getoffer":
cesium.getOffer(args.id) cesium.getOffer(args.id)
elif cmd == "setoffer": elif cmd == "setoffer":
cesium.setOffer(args.title, args.description, args.city, args.localisation, args.category, args.price, args.picture) cesium.setOffer(**cmd_args_dict)
elif cmd == "deleteoffer": elif cmd == "deleteoffer":
cesium.deleteOffer(args.id) cesium.deleteOffer(**cmd_args_dict)
# Construct GVA object
elif cmd in ("pay","history","balance","id","idBalance","currentUd"):
from lib.gva import GvaApi
if args.node:
node = args.node
else: else:
node = os.getenv('NODE') raise ValueError(f"Unknown command: {cmd}")
if not node:
node="https://g1.librelois.fr/gva"
if args.pubkey:
destPubkey = args.pubkey
else:
destPubkey = False
gva = GvaApi(dunikey, node, destPubkey, noNeedDunikey) def handle_gva_commands(args, cmd, gva):
# Get args of the command
cmd_args = (
list(zip(*list(commands[cmd]["arguments"].keys())))[1]
if len(commands[cmd]["arguments"].keys()) > 0
else []
)
cmd_args_dict = {arg: args_dict[arg] for arg in cmd_args if arg in args_dict}
# cmd_args_values = list(cmd_args_dict.values())
if cmd == "pay": if cmd == "pay":
gva.pay(args.amount, args.comment, args.mempool, args.verbose) gva.pay(args.amount, args.comment, args.mempool, args.verbose)
@ -251,12 +454,37 @@ elif cmd in ("pay","history","balance","id","idBalance","currentUd"):
elif cmd == "balance": elif cmd == "balance":
gva.balance(args.mempool) gva.balance(args.mempool)
elif cmd == "id": elif cmd == "id":
gva.id(args.pubkey, args.username) gva.id(**cmd_args_dict)
elif cmd == "idBalance": elif cmd == "idBalance":
gva.idBalance(args.pubkey) gva.idBalance(**cmd_args_dict)
elif cmd == "currentUd": elif cmd == "currentUd":
gva.currentUd() gva.currentUd()
elif cmd == "listWallets":
gva.listWallets(args.brut, args.mbr, args.non_mbr, args.larf)
else:
raise ValueError(f"Unknown command: {cmd}")
# Construct the CesiumPlus object
if commands[cmd]["type"] == "cesium":
if args.node:
pod = args.node
cesium = CesiumPlus(dunikey, pod, noNeedDunikey)
handle_cesium_commands(args, cmd, cesium)
# Construct the GvaApi object
elif commands[cmd]["type"] == "gva":
if args.node:
node = args.node
if hasattr(args, "pubkey"):
destPubkey = args.pubkey
gva = GvaApi(dunikey, node, destPubkey, noNeedDunikey)
handle_gva_commands(args, cmd, gva)
else:
raise ValueError(f"Unknown command: {cmd}")
if keyPath: if keyPath:
os.remove(keyPath) os.remove(keyPath)

View File

@ -1,15 +1,18 @@
import json
import re, string, random, base64 import re, string, random, base64
from lib.cesiumCommon import CesiumCommon, PUBKEY_REGEX from lib.cesiumCommon import CesiumCommon, PUBKEY_REGEX
from lib.geolocProfiles import GeolocProfiles
from lib.getPages import Pages
from lib.messaging import ReadFromCesium, SendToCesium, DeleteFromCesium from lib.messaging import ReadFromCesium, SendToCesium, DeleteFromCesium
from lib.profiles import Profiles from lib.profiles import Profiles
from lib.stars import ReadLikes, SendLikes, UnLikes from lib.stars import ReadLikes, SendLikes, UnLikes
from lib.offers import Offers from lib.offers import Offers
class CesiumPlus(CesiumCommon):
class CesiumPlus(CesiumCommon):
#################### Messaging #################### #################### Messaging ####################
def read(self, nbrMsg, outbox, isJSON): def read(self, nbrMsg, isJSON, outbox):
readCesium = ReadFromCesium(self.dunikey, self.pod) readCesium = ReadFromCesium(self.dunikey, self.pod)
jsonMsg = readCesium.sendDocument(nbrMsg, outbox) jsonMsg = readCesium.sendDocument(nbrMsg, outbox)
if isJSON: if isJSON:
@ -23,12 +26,14 @@ class CesiumPlus(CesiumCommon):
sendCesium.recipient = recipient sendCesium.recipient = recipient
# Generate pseudo-random nonce # Generate pseudo-random nonce
nonce=[] nonce = []
for _ in range(32): for _ in range(32):
nonce.append(random.choice(string.ascii_letters + string.digits)) nonce.append(random.choice(string.ascii_letters + string.digits))
sendCesium.nonce = base64.b64decode(''.join(nonce)) sendCesium.nonce = base64.b64decode("".join(nonce))
finalDoc = sendCesium.configDoc(sendCesium.encryptMsg(title), sendCesium.encryptMsg(msg)) # Configure JSON document to send finalDoc = sendCesium.configDoc(
sendCesium.encryptMsg(title), sendCesium.encryptMsg(msg)
) # Configure JSON document to send
sendCesium.sendDocument(finalDoc, outbox) # Send final signed document sendCesium.sendDocument(finalDoc, outbox) # Send final signed document
def delete(self, idsMsgList, outbox): def delete(self, idsMsgList, outbox):
@ -40,10 +45,21 @@ class CesiumPlus(CesiumCommon):
#################### Profiles #################### #################### Profiles ####################
def set(self, name=None, description=None, ville=None, adresse=None, position=None, site=None, avatar=None): def set(
self,
name=None,
description=None,
ville=None,
adresse=None,
position=None,
site=None,
avatar=None,
):
setProfile = Profiles(self.dunikey, self.pod) setProfile = Profiles(self.dunikey, self.pod)
document = setProfile.configDocSet(name, description, ville, adresse, position, site, avatar) document = setProfile.configDocSet(
result = setProfile.sendDocument(document,'set') name, description, ville, adresse, position, site, avatar
)
result = setProfile.sendDocument(document, "set")
print(result) print(result)
return result return result
@ -53,23 +69,46 @@ class CesiumPlus(CesiumCommon):
if not profile: if not profile:
profile = self.pubkey profile = self.pubkey
if not re.match(PUBKEY_REGEX, profile) or len(profile) > 45: if not re.match(PUBKEY_REGEX, profile) or len(profile) > 45:
scope = 'title' scope = "title"
else: else:
scope = '_id' scope = "_id"
document = getProfile.configDocGet(profile, scope, avatar) document = getProfile.configDocGet(profile, scope, avatar)
resultJSON = getProfile.sendDocument(document, 'get') resultJSON = getProfile.sendDocument(document, "get")
result = getProfile.parseJSON(resultJSON) result = getProfile.parseJSON(resultJSON)
print(result) print(result)
def getPage(self, page=None, avatar=None):
getPage = Pages(self.dunikey, self.pod, self.noNeedDunikey)
if not page:
page = self.pubkey
if not re.match(PUBKEY_REGEX, page) or len(page) > 45:
scope = "title"
else:
scope = "_id"
document = getPage.configDocGet(page, scope, avatar)
resultJSON = getPage.sendDocument(document, "get")
result = getPage.parseJSON(resultJSON)
print(result)
def erase(self): def erase(self):
eraseProfile = Profiles(self.dunikey, self.pod) eraseProfile = Profiles(self.dunikey, self.pod)
document = eraseProfile.configDocErase() document = eraseProfile.configDocErase()
result = eraseProfile.sendDocument(document,'erase') result = eraseProfile.sendDocument(document, "erase")
print(result) print(result)
def geolocProfiles(self, node):
geolocProfiles = GeolocProfiles(self.dunikey, self.pod)
cesiumProfiles = geolocProfiles.getCesiumProfiles()
gvaProfiles = geolocProfiles.getGVAProfiles(node)
result = geolocProfiles.formatProfiles(cesiumProfiles, json.loads(gvaProfiles))
print(json.dumps(result))
#################### Likes #################### #################### Likes ####################
def readLikes(self, profile=False): def readLikes(self, profile=False):
@ -95,10 +134,21 @@ class CesiumPlus(CesiumCommon):
#################### Offer #################### #################### Offer ####################
def setOffer(self, title=None, description=None, city=None, localisation=None, category=None, price=None, picture=None): def setOffer(
self,
title=None,
description=None,
city=None,
location=None,
category=None,
price=None,
picture=None,
):
setOffer = Offers(self.dunikey, self.pod) setOffer = Offers(self.dunikey, self.pod)
document = setOffer.configDocSet(title, description, city, localisation, category, price, picture) document = setOffer.configDocSet(
result = setOffer.sendDocumentSet(document,'set') title, description, city, location, category, price, picture
)
result = setOffer.sendDocumentSet(document, "set")
# print(result) # print(result)
return result return result
@ -106,7 +156,7 @@ class CesiumPlus(CesiumCommon):
def getOffer(self, id, avatar=None): def getOffer(self, id, avatar=None):
getOffer = Offers(self.dunikey, self.pod, self.noNeedDunikey) getOffer = Offers(self.dunikey, self.pod, self.noNeedDunikey)
resultJSON = getOffer.sendDocumentGet(id, 'get') resultJSON = getOffer.sendDocumentGet(id, "get")
# print(resultJSON) # print(resultJSON)
result = getOffer.parseJSON(resultJSON) result = getOffer.parseJSON(resultJSON)
@ -115,6 +165,6 @@ class CesiumPlus(CesiumCommon):
def deleteOffer(self, id): def deleteOffer(self, id):
eraseOffer = Offers(self.dunikey, self.pod) eraseOffer = Offers(self.dunikey, self.pod)
document = eraseOffer.configDocErase(id) document = eraseOffer.configDocErase(id)
result = eraseOffer.sendDocumentSet(document,'delete', id) result = eraseOffer.sendDocumentSet(document, "delete", id)
print(result) print(result)

0
lib/currentUd.py Normal file → Executable file
View File

112
lib/geolocProfiles.py Executable file
View File

@ -0,0 +1,112 @@
import requests
from time import time
from lib.cesiumCommon import CesiumCommon
from lib.gvaWallets import ListWallets
class GeolocProfiles(CesiumCommon):
def getCesiumProfiles(self):
# Send a POST request to the Cesium profiles API
response = requests.post(
"https://g1.data.e-is.pro/user/profile/_search?scroll=2m",
json={
"query": {
"constant_score": {
"filter": [
{"exists": {"field": "geoPoint"}},
{
"geo_bounding_box": {
"geoPoint": {
"top_left": {"lat": 90, "lon": -180},
"bottom_right": {"lat": -90, "lon": 180},
}
}
},
]
}
},
"_source": [
"title",
"avatar._content_type",
"description",
"city",
"address",
"socials.url",
"creationTime",
"membersCount",
"type",
"geoPoint",
],
"size": 20000,
},
)
scroll_id = response.json()["_scroll_id"]
finalResult: dict | None = response.json()["hits"]["hits"]
while True:
# Send a scroll request to get the next page
response_scroll = requests.post(
"https://g1.data.e-is.pro/_search/scroll",
json={"scroll_id": scroll_id, "scroll": "2m"},
)
# Check if the response is empty (no results) or if there's an error
if (
not response_scroll.json()["hits"]["hits"]
or "error" in response_scroll.json()
):
break
else:
finalResult.extend(response_scroll.json()["hits"]["hits"])
# Process the results here
# Delete the scroll context when done
requests.delete(
"https://g1.data.e-is.pro/_search/scroll", json={"scroll_id": [scroll_id]}
)
return finalResult
def getGVAProfiles(self, node):
# Retrieve GVA profiles using the ListWallets class
gva = ListWallets(node, map=True)
return gva.sendDoc()
def formatProfiles(self, cesiumProfiles, gvaProfiles):
walletsResult = []
for profile in cesiumProfiles:
source: dict = profile["_source"]
pubkey: dict = profile["_id"]
if pubkey not in gvaProfiles:
continue
# Extract necessary information from the profiles
id_info: dict = gvaProfiles[pubkey].get("id") or {}
isMember = id_info.get("isMember", False)
userId = id_info.get("username")
title = source.get("title")
city = source.get("city")
avatar = source.get("avatar")
socials = source.get("socials")
description = source.get("description")
address = source.get("address")
walletsResult.append(
{
"pubkey": pubkey,
**({"address": address} if address else {}),
**({"city": city} if city else {}),
**({"description": description} if description else {}),
**({"avatar": avatar} if avatar else {}),
**({"userId": userId} if userId else {}),
"isMember": isMember,
"geoPoint": source["geoPoint"],
**({"socials": socials} if socials else {}),
**({"title": title} if title else {}),
}
)
return {"wallets": walletsResult, "time": int(time())}

125
lib/getPages.py Executable file
View File

@ -0,0 +1,125 @@
import sys, re, json, requests, base64
from time import time
from lib.cesiumCommon import CesiumCommon, PUBKEY_REGEX
class Pages(CesiumCommon):
# Configure JSON document SET to send
def configDocSet(self, name, description, city, address, pos, socials, avatar):
timeSent = int(time())
data = {}
if name: data['title'] = name
if description: data['description'] = description
if address: data['address'] = address
if city: data['city'] = city
if pos:
geoPoint = {}
geoPoint['lat'] = pos[0]
geoPoint['lon'] = pos[1]
data['geoPoint'] = geoPoint
if socials:
data['socials'] = []
data['socials'].append({})
data['socials'][0]['type'] = "web"
data['socials'][0]['url'] = socials
if avatar:
avatar = open(avatar, 'rb').read()
avatar = base64.b64encode(avatar).decode()
data['avatar'] = {}
data['avatar']['_content'] = avatar
data['avatar']['_content_type'] = "image/png"
data['time'] = timeSent
data['issuer'] = self.pubkey
data['version'] = 2
data['tags'] = []
document = json.dumps(data)
return self.signDoc(document)
# Configure JSON document GET to send
def configDocGet(self, profile, scope='title', getAvatar=None):
if getAvatar:
avatar = "avatar"
else:
avatar = "avatar._content_type"
data = {
"query": {
"bool": {
"should":[
{
"match":{
scope:{
"query": profile,"boost":2
}
}
},{
"prefix": {scope: profile}
}
]
}
},"highlight": {
"fields": {
"title":{},
"tags":{}
}
},"from":0,
"size":100,
"_source":["title", avatar,"description","city","address","socials.url","creationTime","membersCount","type","geoPoint"],
"indices_boost":{"user":100,"page":1,"group":0.01
}
}
document = json.dumps(data)
return document
# Configure JSON document SET to send
def configDocErase(self):
timeSent = int(time())
data = {}
data['time'] = timeSent
data['id'] = self.pubkey
data['issuer'] = self.pubkey
data['version'] = 2
data['index'] = "user"
data['type'] = "profile"
document = json.dumps(data)
return self.signDoc(document)
def sendDocument(self, document, type):
headers = {
'Content-type': 'application/json',
}
# Send JSON document and get JSON result
if type == 'set':
reqQuery = '{0}/user/profile?pubkey={1}/_update?pubkey={1}'.format(self.pod, self.pubkey)
elif type == 'get':
reqQuery = '{0}/user,page,group/profile,record/_search'.format(self.pod)
elif type == 'erase':
reqQuery = '{0}/history/delete'.format(self.pod)
result = requests.post(reqQuery, headers=headers, data=document)
if result.status_code == 200:
# print(result.text)
return result.text
else:
sys.stderr.write("Echec de l'envoi du document...\n" + result.text + '\n')
def parseJSON(self, doc):
doc = json.loads(doc)['hits']['hits']
if doc:
pubkey = { "pubkey": doc[0]['_id'] }
rest = doc[0]['_source']
final = {**pubkey, **rest}
return json.dumps(final, indent=2)
else:
return 'Profile vide'

View File

@ -1,4 +1,5 @@
from lib.currentUd import currentUd from lib.currentUd import currentUd
from lib.gvaWallets import ListWallets
import sys, re import sys, re
from lib.natools import get_privkey from lib.natools import get_privkey
from lib.gvaPay import Transaction, PUBKEY_REGEX from lib.gvaPay import Transaction, PUBKEY_REGEX
@ -38,6 +39,7 @@ class GvaApi():
#################### Payments #################### #################### Payments ####################
def pay(self, amount, comment, mempool, verbose): def pay(self, amount, comment, mempool, verbose):
comment = " ".join(comment)
gva = Transaction(self.dunikey, self.node, self.destPubkey, amount, comment, mempool, verbose) gva = Transaction(self.dunikey, self.node, self.destPubkey, amount, comment, mempool, verbose)
gva.genDoc() gva.genDoc()
gva.checkTXDoc() gva.checkTXDoc()
@ -74,3 +76,9 @@ class GvaApi():
gva = currentUd(self.node) gva = currentUd(self.node)
result = gva.sendDoc() result = gva.sendDoc()
print(result) print(result)
def listWallets(self, brut, brutMbr, brutNonMbr, brutLarf):
gva = ListWallets(self.node, brut, brutMbr, brutNonMbr, brutLarf)
result = gva.sendDoc()
print(result)

View File

@ -43,9 +43,11 @@ class History:
node { node {
currency currency
issuers issuers
blockstamp
outputs outputs
comment comment
writtenTime writtenTime
hash
} }
} }
} }
@ -57,13 +59,17 @@ class History:
comment comment
outputs outputs
receivedTime receivedTime
blockstamp
hash
} }
receiving { sending {
currency currency
issuers issuers
comment comment
outputs outputs
receivedTime receivedTime
blockstamp
hash
} }
} }
balance(script: $script) { balance(script: $script) {
@ -119,6 +125,7 @@ class History:
trans[i] = [] trans[i] = []
trans[i].append(direction) trans[i].append(direction)
trans[i].append(transaction['writtenTime']) trans[i].append(transaction['writtenTime'])
if direction == 'SENT': if direction == 'SENT':
trans[i].append(outPubkey) trans[i].append(outPubkey)
amount = int('-' + output.split(':')[0]) amount = int('-' + output.split(':')[0])
@ -133,12 +140,12 @@ class History:
trans[i].append(round(amount/self.UD, 2)) trans[i].append(round(amount/self.UD, 2))
trans[i].append(transaction['comment']) trans[i].append(transaction['comment'])
trans[i].append(base) trans[i].append(base)
trans[i].append(transaction['blockstamp'])
trans[i].append(transaction['hash'])
i += 1 i += 1
# Parse transactions in mempool # Parse transactions in mempool
for direction in self.historyDoc['txsHistoryMp']: for direction, resBc in self.historyDoc['txsHistoryMp'].items():
resBc = []
resBc = self.historyDoc['txsHistoryMp'][direction]
for j, transaction in enumerate(resBc): for j, transaction in enumerate(resBc):
# print(transaction) # print(transaction)
transaction = resBc[j] transaction = resBc[j]
@ -147,9 +154,11 @@ class History:
# if direction == 'RECEIVING' or self.pubkey != outPubkey: # if direction == 'RECEIVING' or self.pubkey != outPubkey:
trans.append(i) trans.append(i)
trans[i] = [] trans[i] = []
trans[i].append(direction) trans[i].append(direction.upper())
trans[i].append(int(time.time())) trans[i].append(transaction['receivedTime'])
if direction == 'SENDING':
# trans[i].append(int(time.time()))
if direction.upper() == 'SENDING':
trans[i].append(outPubkey) trans[i].append(outPubkey)
amount = int('-' + output.split(':')[0]) amount = int('-' + output.split(':')[0])
else: else:
@ -163,6 +172,8 @@ class History:
trans[i].append(round(amount/self.UD, 2)) trans[i].append(round(amount/self.UD, 2))
trans[i].append(transaction['comment']) trans[i].append(transaction['comment'])
trans[i].append(base) trans[i].append(base)
trans[i].append(transaction['blockstamp'])
trans[i].append(transaction['hash'])
i += 1 i += 1
# Order transactions by date # Order transactions by date
@ -257,11 +268,14 @@ class History:
for i, trans in enumerate(transList): for i, trans in enumerate(transList):
dailyJSON.append(i) dailyJSON.append(i)
dailyJSON[i] = {} dailyJSON[i] = {}
dailyJSON[i]['status'] = trans[0].upper()
dailyJSON[i]['date'] = trans[1] dailyJSON[i]['date'] = trans[1]
dailyJSON[i]['pubkey'] = trans[2] dailyJSON[i]['pubkey'] = trans[2]
dailyJSON[i]['amount'] = trans[3] dailyJSON[i]['amount'] = trans[3]
dailyJSON[i]['amountUD'] = trans[4] dailyJSON[i]['amountUD'] = trans[4]
dailyJSON[i]['comment'] = trans[5] dailyJSON[i]['comment'] = trans[5]
dailyJSON[i]['blockstamp'] = trans[7]
dailyJSON[i]['hash'] = trans[8]
dailyJSON = json.dumps(dailyJSON, indent=2) dailyJSON = json.dumps(dailyJSON, indent=2)
# If we want to write JSON to a file # If we want to write JSON to a file

0
lib/gvaID.py Normal file → Executable file
View File

100
lib/gvaWallets.py Executable file
View File

@ -0,0 +1,100 @@
#!/usr/bin/env python3
import sys
import json
from gql import gql, Client
from gql.transport.aiohttp import AIOHTTPTransport
from lib.natools import fmt, sign, get_privkey
class ListWallets:
def __init__(
self, node=False, brut=False, mbr=False, nonMbr=False, larf=False, map=False
):
# Initialize the ListWallets class with optional filters
self.mbr = mbr # Filter for members
self.larf = larf # Filter for non-empty identities
self.nonMbr = nonMbr # Filter for non-members
self.brut = brut # Output format flag (brut or JSON)
self.map = map # Output format flag (map or list)
# Define Duniter GVA node
transport = AIOHTTPTransport(url=node)
self.client = Client(transport=transport, fetch_schema_from_transport=True)
def sendDoc(self):
# Define the GraphQL query to retrieve wallet information
queryBuild = gql(
"""
{
wallets(pagination: { cursor: null, ord: ASC, pageSize: 0 }) {
pageInfo {
hasNextPage
endCursor
}
edges {
node {
script
balance {
amount
base
}
idty {
isMember
username
}
}
}
}
}
"""
)
try:
# Execute the GraphQL query
queryResult = self.client.execute(queryBuild)
except Exception as e:
# Handle any exceptions that occur during the query
sys.stderr.write("Failed to retrieve the list:\n" + str(e) + "\n")
sys.exit(1)
jsonBrut = queryResult["wallets"]["edges"]
walletList = []
walletMap = {}
for i, trans in enumerate(jsonBrut):
dataWork = trans["node"]
identity = dataWork["idty"]
is_member = identity and identity["isMember"]
# Apply filters based on member status and larf flag
member_filter = self.mbr and not is_member
non_member_filter = self.nonMbr and is_member
larf_filter = self.larf and identity
if member_filter or non_member_filter or larf_filter:
continue
wallet_data = {
"pubkey": dataWork["script"],
"balance": dataWork["balance"]["amount"] / 100,
"id": identity,
}
if self.map:
walletMap[dataWork["script"]] = wallet_data
else:
walletList.append(wallet_data)
if self.brut:
# Generate a list of formatted wallet names using list comprehension
names = [
wallet["pubkey"]
if not (self.mbr or self.nonMbr) or wallet["id"] is None
else f'{wallet["pubkey"]} {wallet["id"]["username"]}'
for wallet in walletList
]
return "\n".join(names)
else:
# Return JSON data in either map or list format
return json.dumps(walletMap if self.map else walletList, indent=2)

View File

@ -46,6 +46,7 @@ class ReadFromCesium(CesiumCommon):
# Parse JSON result and display messages # Parse JSON result and display messages
def readMessages(self, msgJSON, nbrMsg, outbox): def readMessages(self, msgJSON, nbrMsg, outbox):
def decrypt(msg): def decrypt(msg):
if msg is None: return ''
msg64 = base64.b64decode(msg) msg64 = base64.b64decode(msg)
return box_decrypt(msg64, get_privkey(self.dunikey, "pubsec"), self.issuer, nonce).decode() return box_decrypt(msg64, get_privkey(self.dunikey, "pubsec"), self.issuer, nonce).decode()
@ -67,7 +68,10 @@ class ReadFromCesium(CesiumCommon):
msgSrc = hits["_source"] msgSrc = hits["_source"]
self.issuer = msgSrc["issuer"] self.issuer = msgSrc["issuer"]
nonce = msgSrc["nonce"] nonce = msgSrc["nonce"]
try:
nonce = base58.b58decode(nonce) nonce = base58.b58decode(nonce)
except:
nonce = base58.b58decode('5aZdSqKGHBqm2uMPwN6XnfiiJKRieb1Hh')
self.dateS = msgSrc["time"] self.dateS = msgSrc["time"]
date = datetime.fromtimestamp(self.dateS).strftime(", le %d/%m/%Y à %H:%M ") date = datetime.fromtimestamp(self.dateS).strftime(", le %d/%m/%Y à %H:%M ")
if outbox: if outbox:
@ -94,6 +98,7 @@ class ReadFromCesium(CesiumCommon):
# Parse JSON result and display messages # Parse JSON result and display messages
def jsonMessages(self, msgJSON, nbrMsg, outbox): def jsonMessages(self, msgJSON, nbrMsg, outbox):
def decrypt(msg): def decrypt(msg):
if msg is None: return ''
msg64 = base64.b64decode(msg) msg64 = base64.b64decode(msg)
return box_decrypt(msg64, get_privkey(self.dunikey, "pubsec"), self.issuer, nonce).decode() return box_decrypt(msg64, get_privkey(self.dunikey, "pubsec"), self.issuer, nonce).decode()
@ -113,7 +118,10 @@ class ReadFromCesium(CesiumCommon):
msgSrc = hits["_source"] msgSrc = hits["_source"]
self.issuer = msgSrc["issuer"] self.issuer = msgSrc["issuer"]
nonce = msgSrc["nonce"] nonce = msgSrc["nonce"]
try:
nonce = base58.b58decode(nonce) nonce = base58.b58decode(nonce)
except:
nonce = base58.b58decode('5aZdSqKGHBqm2uMPwN6XnfiiJKRieb1Hh')
self.date = msgSrc["time"] self.date = msgSrc["time"]
if outbox: if outbox:
@ -123,11 +131,18 @@ class ReadFromCesium(CesiumCommon):
try: try:
self.title = decrypt(msgSrc["title"]) self.title = decrypt(msgSrc["title"])
except Exception as e:
self.title = "jaklis can't read that mother fucker"
# sys.stderr.write(colored(str('Error decrypt message title: ' + str(e)), 'red') + '\n')
# pp_json(hits)
try:
self.content = decrypt(msgSrc["content"]) self.content = decrypt(msgSrc["content"])
except Exception as e: except Exception as e:
sys.stderr.write(colored(str(e), 'red') + '\n') self.content = "jaklis can't read that mother fucker"
pp_json(hits) # sys.stderr.write(colored(str('Error decrypt message content: ' + str(e)), 'red') + '\n')
continue # pp_json(hits)
data.append(i) data.append(i)
data[i] = {} data[i] = {}
@ -136,6 +151,7 @@ class ReadFromCesium(CesiumCommon):
data[i]['pubkey'] = pubkey data[i]['pubkey'] = pubkey
data[i]['title'] = self.title data[i]['title'] = self.title
data[i]['content'] = self.content data[i]['content'] = self.content
# print('toto')
data = json.dumps(data, indent=2) data = json.dumps(data, indent=2)
return data return data

8
lib/offers.py Normal file → Executable file
View File

@ -4,7 +4,7 @@ from lib.cesiumCommon import CesiumCommon, PUBKEY_REGEX
class Offers(CesiumCommon): class Offers(CesiumCommon):
# Configure JSON document SET to send # Configure JSON document SET to send
def configDocSet(self, title, description, city, localisation, category, price: float, picture): def configDocSet(self, title, description, city, location, category, price: float, picture):
timeSent = int(time()) timeSent = int(time())
# {"parent":"cat90","localizedNames":{"en":"Fruits & Vegetables","es-ES":"Frutas y Vegetales","fr-FR":"Fruits & Légumes"},"name":"Fruits & Légumes","id":"cat92"} # {"parent":"cat90","localizedNames":{"en":"Fruits & Vegetables","es-ES":"Frutas y Vegetales","fr-FR":"Fruits & Légumes"},"name":"Fruits & Légumes","id":"cat92"}
@ -13,10 +13,10 @@ class Offers(CesiumCommon):
if title: data['title'] = title if title: data['title'] = title
if description: data['description'] = description if description: data['description'] = description
if city: data['city'] = city if city: data['city'] = city
if localisation: if location:
geoPoint = {} geoPoint = {}
geoPoint['lat'] = localisation[0] geoPoint['lat'] = location[0]
geoPoint['lon'] = localisation[1] geoPoint['lon'] = location[1]
data['geoPoint'] = geoPoint data['geoPoint'] = geoPoint
if picture: if picture:
picture = open(picture, 'rb').read() picture = open(picture, 'rb').read()

View File

@ -1,4 +1,4 @@
import sys, re, json, requests, base64 import sys, json, requests, base64
from time import time from time import time
from lib.cesiumCommon import CesiumCommon, PUBKEY_REGEX from lib.cesiumCommon import CesiumCommon, PUBKEY_REGEX
@ -9,38 +9,41 @@ class Profiles(CesiumCommon):
timeSent = int(time()) timeSent = int(time())
data = {} data = {}
if name: data['title'] = name if name:
if description: data['description'] = description data["title"] = name
if address: data['address'] = address if description:
if city: data['city'] = city data["description"] = description
if address:
data["address"] = address
if city:
data["city"] = city
if pos: if pos:
geoPoint = {} geoPoint = {}
geoPoint['lat'] = pos[0] geoPoint["lat"] = pos[0]
geoPoint['lon'] = pos[1] geoPoint["lon"] = pos[1]
data['geoPoint'] = geoPoint data["geoPoint"] = geoPoint
if socials: if socials:
data['socials'] = [] data["socials"] = []
data['socials'].append({}) data["socials"].append({})
data['socials'][0]['type'] = "web" data["socials"][0]["type"] = "web"
data['socials'][0]['url'] = socials data["socials"][0]["url"] = socials
if avatar: if avatar:
avatar = open(avatar, 'rb').read() avatar = open(avatar, "rb").read()
avatar = base64.b64encode(avatar).decode() avatar = base64.b64encode(avatar).decode()
data['avatar'] = {} data["avatar"] = {}
data['avatar']['_content'] = avatar data["avatar"]["_content"] = avatar
data['avatar']['_content_type'] = "image/png" data["avatar"]["_content_type"] = "image/png"
data['time'] = timeSent data["time"] = timeSent
data['issuer'] = self.pubkey data["issuer"] = self.pubkey
data['version'] = 2 data["version"] = 2
data['tags'] = [] data["tags"] = []
document = json.dumps(data) document = json.dumps(data)
return self.signDoc(document) return self.signDoc(document)
# Configure JSON document GET to send # Configure JSON document GET to send
def configDocGet(self, profile, scope='title', getAvatar=None): def configDocGet(self, profile, scope="title", getAvatar=None):
if getAvatar: if getAvatar:
avatar = "avatar" avatar = "avatar"
else: else:
@ -49,28 +52,28 @@ class Profiles(CesiumCommon):
data = { data = {
"query": { "query": {
"bool": { "bool": {
"should":[ "should": [
{ {"match": {scope: {"query": profile, "boost": 2}}},
"match":{ {"prefix": {scope: profile}},
scope:{
"query": profile,"boost":2
}
}
},{
"prefix": {scope: profile}
}
] ]
} }
},"highlight": { },
"fields": { "highlight": {"fields": {"title": {}, "tags": {}}},
"title":{}, "from": 0,
"tags":{} "size": 100,
} "_source": [
},"from":0, "title",
"size":100, avatar,
"_source":["title", avatar,"description","city","address","socials.url","creationTime","membersCount","type","geoPoint"], "description",
"indices_boost":{"user":100,"page":1,"group":0.01 "city",
} "address",
"socials.url",
"creationTime",
"membersCount",
"type",
"geoPoint",
],
"indices_boost": {"user": 100, "page": 1, "group": 0.01},
} }
document = json.dumps(data) document = json.dumps(data)
@ -82,44 +85,45 @@ class Profiles(CesiumCommon):
timeSent = int(time()) timeSent = int(time())
data = {} data = {}
data['time'] = timeSent data["time"] = timeSent
data['id'] = self.pubkey data["id"] = self.pubkey
data['issuer'] = self.pubkey data["issuer"] = self.pubkey
data['version'] = 2 data["version"] = 2
data['index'] = "user" data["index"] = "user"
data['type'] = "profile" data["type"] = "profile"
document = json.dumps(data) document = json.dumps(data)
return self.signDoc(document) return self.signDoc(document)
def sendDocument(self, document, type): def sendDocument(self, document, type):
headers = { headers = {
'Content-type': 'application/json', "Content-type": "application/json",
} }
# Send JSON document and get JSON result # Send JSON document and get JSON result
if type == 'set': if type == "set":
reqQuery = '{0}/user/profile?pubkey={1}/_update?pubkey={1}'.format(self.pod, self.pubkey) reqQuery = "{0}/user/profile?pubkey={1}/_update?pubkey={1}".format(
elif type == 'get': self.pod, self.pubkey
reqQuery = '{0}/user,page,group/profile,record/_search'.format(self.pod) )
elif type == 'erase': elif type == "get":
reqQuery = '{0}/history/delete'.format(self.pod) reqQuery = "{0}/user,page,group/profile,record/_search".format(self.pod)
elif type == "erase":
reqQuery = "{0}/history/delete".format(self.pod)
result = requests.post(reqQuery, headers=headers, data=document) result = requests.post(reqQuery, headers=headers, data=document)
if result.status_code == 200: if result.status_code == 200:
# print(result.text) # print(result.text)
return result.text return result.text
else: else:
sys.stderr.write("Echec de l'envoi du document...\n" + result.text + '\n') sys.stderr.write("Echec de l'envoi du document...\n" + result.text + "\n")
def parseJSON(self, doc): def parseJSON(self, doc):
doc = json.loads(doc)['hits']['hits'] doc = json.loads(doc)["hits"]["hits"]
if doc: if doc:
pubkey = { "pubkey": doc[0]['_id'] } pubkey = {"pubkey": doc[0]["_id"]}
rest = doc[0]['_source'] rest = doc[0]["_source"]
final = {**pubkey, **rest} final = {**pubkey, **rest}
return json.dumps(final, indent=2) return json.dumps(final, indent=2)
else: else:
return 'Profile vide' return "Profile vide"

View File

@ -1,9 +1,8 @@
wheel wheel
base58 base58
pybase64 pybase64
duniterpy==0.62.0 duniterpy
termcolor termcolor
python-dotenv python-dotenv
gql==3.0.0a5 gql
#gql==2.0
requests requests

View File

@ -1,5 +1,7 @@
#!/bin/bash #!/bin/bash
hasError=0
for i in gcc python3-pip python3-setuptools libpq-dev python3-dev python3-wheel; do for i in gcc python3-pip python3-setuptools libpq-dev python3-dev python3-wheel; do
if [ $(dpkg-query -W -f='${Status}' $i 2>/dev/null | grep -c "ok installed") -eq 0 ]; then if [ $(dpkg-query -W -f='${Status}' $i 2>/dev/null | grep -c "ok installed") -eq 0 ]; then
[[ ! $j ]] && sudo apt update [[ ! $j ]] && sudo apt update
@ -8,5 +10,12 @@ for i in gcc python3-pip python3-setuptools libpq-dev python3-dev python3-wheel;
fi fi
done done
pip3 install -r requirements.txt pip3 install -r requirements.txt || hasError=1
chmod u+x jaklis.py chmod u+x jaklis.py
sudo ln -sf $(realpath jaklis.py) /usr/local/bin/jaklis || hasError=1
if [[ hasError -eq 0 ]]; then
echo "Setup done. You can use 'jaklis' command, try it."
else
echo "An error has occurred"
fi