Compare commits

...

18 Commits
0.5.1 ... 0.5.6

Author SHA1 Message Date
Mark Qvist
d8cfc69ac6 Updated dependencies 2024-12-12 08:54:59 +01:00
Mark Qvist
ccc41a5789 Updated version 2024-12-12 08:54:05 +01:00
Mark Qvist
7b38d4f80e Fix multiline prop 2024-12-12 08:53:20 +01:00
Mark Qvist
bec7612428 Updated dependencies 2024-12-09 22:49:55 +01:00
Mark Qvist
11fd305959 Updated versions 2024-12-09 22:11:49 +01:00
markqvist
c2fc7039fd Merge pull request #62 from RFnexus/micron
Add Checkbox and Radio Group fields to Micron
2024-11-27 14:15:02 +01:00
zenith
912c510ab2 Add Checkbox and Radio Button fields to Micron format 2024-11-26 19:58:53 -05:00
probability
a5aa2097bd Add Checkbox and Radio Button fields to Micron format 2024-11-25 18:18:33 -05:00
Mark Qvist
dc43bc6a22 Fix invalid LXMF link handling in browser 2024-11-15 16:29:46 +01:00
Mark Qvist
289136a632 Updated version 2024-10-06 11:26:05 +02:00
Mark Qvist
0e79c3299c Added opportunistic delivery if destination ratchets are available 2024-10-06 11:25:08 +02:00
Mark Qvist
c61da069f2 Updated dependencies 2024-10-06 11:14:17 +02:00
Mark Qvist
0df8b56d58 Updated versions 2024-09-17 14:53:15 +02:00
Mark Qvist
112a45f270 Fixed invalid dict key. Fixes #59. 2024-09-17 14:50:32 +02:00
Mark Qvist
03a02a9ebc Fixed incorrect display name loading in conversation list 2024-09-11 13:02:05 +02:00
Mark Qvist
e755641dbe Updated version 2024-09-11 11:49:50 +02:00
Mark Qvist
5392275782 Updated dependencies 2024-09-11 11:49:36 +02:00
Mark Qvist
1bbfacee94 Add stamp configuration options 2024-09-11 00:20:35 +02:00
8 changed files with 289 additions and 67 deletions

View File

@@ -102,7 +102,7 @@ class Conversation:
unread = True unread = True
if display_name == None and app_data: if display_name == None and app_data:
display_name = app_data.decode("utf-8") display_name = LXMF.display_name_from_app_data(app_data)
if display_name == None: if display_name == None:
sort_name = "" sort_name = ""
@@ -216,8 +216,16 @@ class Conversation:
if self.app.directory.preferred_delivery(dest.hash) == DirectoryEntry.PROPAGATED: if self.app.directory.preferred_delivery(dest.hash) == DirectoryEntry.PROPAGATED:
if self.app.message_router.get_outbound_propagation_node() != None: if self.app.message_router.get_outbound_propagation_node() != None:
desired_method = LXMF.LXMessage.PROPAGATED desired_method = LXMF.LXMessage.PROPAGATED
else:
if not self.app.message_router.delivery_link_available(dest.hash) and RNS.Identity.current_ratchet_id(dest.hash) != None:
RNS.log(f"Have ratchet for {RNS.prettyhexrep(dest.hash)}, requesting opportunistic delivery of message", RNS.LOG_DEBUG)
desired_method = LXMF.LXMessage.OPPORTUNISTIC
lxm = LXMF.LXMessage(dest, source, content, title=title, desired_method=desired_method) dest_is_trusted = False
if self.app.directory.trust_level(dest.hash) == DirectoryEntry.TRUSTED:
dest_is_trusted = True
lxm = LXMF.LXMessage(dest, source, content, title=title, desired_method=desired_method, include_ticket=dest_is_trusted)
lxm.register_delivery_callback(self.message_notification) lxm.register_delivery_callback(self.message_notification)
lxm.register_failed_callback(self.message_notification) lxm.register_failed_callback(self.message_notification)

View File

@@ -134,6 +134,10 @@ class NomadNetworkApp:
self.lxmf_sync_interval = 360*60 self.lxmf_sync_interval = 360*60
self.lxmf_sync_limit = 8 self.lxmf_sync_limit = 8
self.compact_stream = False self.compact_stream = False
self.required_stamp_cost = None
self.accept_invalid_stamps = False
if not os.path.isdir(self.storagepath): if not os.path.isdir(self.storagepath):
os.makedirs(self.storagepath) os.makedirs(self.storagepath)
@@ -296,8 +300,9 @@ class NomadNetworkApp:
for destination_hash in self.ignored_list: for destination_hash in self.ignored_list:
self.message_router.ignore_destination(destination_hash) self.message_router.ignore_destination(destination_hash)
self.lxmf_destination = self.message_router.register_delivery_identity(self.identity, display_name=self.peer_settings["display_name"]) self.lxmf_destination = self.message_router.register_delivery_identity(self.identity, display_name=self.peer_settings["display_name"], stamp_cost=self.required_stamp_cost)
self.lxmf_destination.set_default_app_data(self.get_display_name_bytes) if not self.accept_invalid_stamps:
self.message_router.enforce_stamps()
RNS.Identity.remember( RNS.Identity.remember(
packet_hash=None, packet_hash=None,
@@ -492,7 +497,9 @@ class NomadNetworkApp:
self.message_router.cancel_propagation_node_requests() self.message_router.cancel_propagation_node_requests()
def announce_now(self): def announce_now(self):
self.lxmf_destination.announce() self.message_router.set_inbound_stamp_cost(self.lxmf_destination.hash, self.required_stamp_cost)
self.lxmf_destination.display_name = self.peer_settings["display_name"]
self.message_router.announce(self.lxmf_destination.hash)
self.peer_settings["last_announce"] = time.time() self.peer_settings["last_announce"] = time.time()
self.save_peer_settings() self.save_peer_settings()
@@ -738,6 +745,24 @@ class NomadNetworkApp:
else: else:
self.lxmf_sync_limit = None self.lxmf_sync_limit = None
if option == "required_stamp_cost":
value = self.config["client"][option]
if value.lower() == "none":
self.required_stamp_cost = None
else:
value = self.config["client"].as_int(option)
if value > 0:
if value > 255:
value = 255
self.required_stamp_cost = value
else:
self.required_stamp_cost = None
if option == "accept_invalid_stamps":
value = self.config["client"].as_bool(option)
self.accept_invalid_stamps = value
if option == "max_accepted_size": if option == "max_accepted_size":
value = self.config["client"].as_float(option) value = self.config["client"].as_float(option)
@@ -1017,6 +1042,24 @@ lxmf_sync_interval = 360
# the limit, and download everything every time. # the limit, and download everything every time.
lxmf_sync_limit = 8 lxmf_sync_limit = 8
# You can specify a required stamp cost for
# inbound messages to be accepted. Specifying
# a stamp cost will require untrusted senders
# that message you to include a cryptographic
# stamp in their messages. Performing this
# operation takes the sender an amount of time
# proportional to the stamp cost. As a rough
# estimate, a stamp cost of 8 will take less
# than a second to compute, and a stamp cost
# of 20 could take several minutes, even on
# a fast computer.
required_stamp_cost = None
# You can signal stamp requirements to senders,
# but still accept messages with invalid stamps
# by setting this option to True.
accept_invalid_stamps = False
# The maximum accepted unpacked size for mes- # The maximum accepted unpacked size for mes-
# sages received directly from other peers, # sages received directly from other peers,
# specified in kilobytes. Messages larger than # specified in kilobytes. Messages larger than

View File

@@ -1 +1 @@
__version__ = "0.5.1" __version__ = "0.5.6"

View File

@@ -17,6 +17,8 @@ The following section contains a simple set of fields, and a few different links
-= -=
>>>Text Fields
An input field : `B444`<username`Entered data>`b An input field : `B444`<username`Entered data>`b
An masked field : `B444`<!|password`Value of Field>`b An masked field : `B444`<!|password`Value of Field>`b
@@ -27,7 +29,24 @@ Two fields : `B444`<8|one`One>`b `B444`<8|two`Two>`b
The data can be `!`[submitted`:/page/input_fields.mu`username|two]`!. The data can be `!`[submitted`:/page/input_fields.mu`username|two]`!.
You can `!`[submit`:/page/input_fields.mu`one|password|small]`! other fields, or just `!`[a single one`:/page/input_fields.mu`username]`! >> Checkbox Fields
`B444`<?|sign_up|1|*`>`b Sign me up
>> Radio group
Select your favorite color:
`B900`<^|color|Red`>`b Red
`B090`<^|color|Green`>`b Green
`B009`<^|color|Blue`>`b Blue
>>> Submitting data
You can `!`[submit`:/page/input_fields.mu`one|password|small|color]`! other fields, or just `!`[a single one`:/page/input_fields.mu`username]`!
Or simply `!`[submit them all`:/page/input_fields.mu`*]`!. Or simply `!`[submit them all`:/page/input_fields.mu`*]`!.

View File

@@ -1,4 +1,5 @@
import RNS import RNS
import LXMF
import os import os
import time import time
import urwid import urwid
@@ -179,23 +180,47 @@ class Browser:
else: else:
link_fields.append(e) link_fields.append(e)
def recurse_down(w): def recurse_down(w):
target = None if isinstance(w, list):
if isinstance(w, list): for t in w:
for t in w: recurse_down(t)
recurse_down(t) elif isinstance(w, tuple):
elif isinstance(w, tuple): for t in w:
for t in w: recurse_down(t)
recurse_down(t) elif hasattr(w, "contents"):
elif hasattr(w, "contents"): recurse_down(w.contents)
recurse_down(w.contents) elif hasattr(w, "original_widget"):
elif hasattr(w, "original_widget"): recurse_down(w.original_widget)
recurse_down(w.original_widget) elif hasattr(w, "_original_widget"):
elif hasattr(w, "_original_widget"): recurse_down(w._original_widget)
recurse_down(w._original_widget) else:
else: if hasattr(w, "field_name") and (all_fields or w.field_name in link_fields):
if hasattr(w, "field_name") and (all_fields or w.field_name in link_fields): field_key = "field_" + w.field_name
request_data["field_"+w.field_name] = w.get_edit_text() if isinstance(w, urwid.Edit):
request_data[field_key] = w.edit_text
elif isinstance(w, urwid.RadioButton):
if w.state:
user_data = getattr(w, "field_value", None)
if user_data is not None:
request_data[field_key] = user_data
elif isinstance(w, urwid.CheckBox):
user_data = getattr(w, "field_value", "1")
if w.state:
existing_value = request_data.get(field_key, '')
if existing_value:
# Concatenate the new value with the existing one
request_data[field_key] = existing_value + ',' + user_data
else:
# Initialize the field with the current value
request_data[field_key] = user_data
else:
pass # do nothing if checkbox is not check
recurse_down(self.attr_maps) recurse_down(self.attr_maps)
RNS.log("Including request data: "+str(request_data), RNS.LOG_DEBUG) RNS.log("Including request data: "+str(request_data), RNS.LOG_DEBUG)
@@ -252,7 +277,7 @@ class Browser:
display_name = None display_name = None
if display_name_data != None: if display_name_data != None:
display_name = display_name_data.decode("utf-8") display_name = LXMF.display_name_from_app_data(display_name_data)
if not source_hash_text in [c[0] for c in existing_conversations]: if not source_hash_text in [c[0] for c in existing_conversations]:
entry = DirectoryEntry(bytes.fromhex(source_hash_text), display_name=display_name) entry = DirectoryEntry(bytes.fromhex(source_hash_text), display_name=display_name)

View File

@@ -521,6 +521,18 @@ The number of minutes between each automatic sync. The default is equal to 6 hou
On low-bandwidth networks, it can be useful to limit the amount of messages downloaded in each sync. The default is 8. Set to 0 to download all available messages every time a sync occurs. On low-bandwidth networks, it can be useful to limit the amount of messages downloaded in each sync. The default is 8. Set to 0 to download all available messages every time a sync occurs.
< <
>>>
`!required_stamp_cost = None`!
>>>>
You can specify a required stamp cost for inbound messages to be accepted. Specifying a stamp cost will require untrusted senders that message you to include a cryptographic stamp in their messages. Performing this operation takes the sender an amount of time proportional to the stamp cost. As a rough estimate, a stamp cost of 8 will take less than a second to compute, and a stamp cost of 20 could take several minutes, even on a fast computer.
<
>>>
`!accept_invalid_stamps = False`!
>>>>
You can signal stamp requirements to senders, but still accept messages with invalid stamps by setting this option to True.
<
>>> >>>
`!max_accepted_size = 500`! `!max_accepted_size = 500`!
>>>> >>>>
@@ -739,12 +751,11 @@ If you have Internet access, and just want to get started experimenting, you are
The Testnet also runs the latest version of Reticulum, often even a short while before it is publicly released, which means strange behaviour might occur. If none of that scares you, add the following interface to your Reticulum configuration file to join: The Testnet also runs the latest version of Reticulum, often even a short while before it is publicly released, which means strange behaviour might occur. If none of that scares you, add the following interface to your Reticulum configuration file to join:
>> >>
[[RNS Testnet Zurich]] [[RNS Testnet Dublin]]
type = TCPClientInterface type = TCPClientInterface
interface_enabled = yes enabled = yes
outgoing = True target_host = dublin.connect.reticulum.network
target_host = zurich.connect.reticulum.network target_port = 4965
target_port = 4242
< <
If you connect to the testnet, you can leave nomadnet running for a while and wait for it to receive announces from other nodes on the network that host pages or services, or you can try connecting directly to some nodes listed here: If you connect to the testnet, you can leave nomadnet running for a while and wait for it to receive announces from other nodes on the network that host pages or services, or you can try connecting directly to some nodes listed here:
@@ -1141,6 +1152,47 @@ A sized input field: `B444`<16|with_size`>`B333
A masked input field: `B444`<!|masked_demo`hidden text>`B333 A masked input field: `B444`<!|masked_demo`hidden text>`B333
Full control: `B444`<!32|all_options`hidden text>`B333 Full control: `B444`<!32|all_options`hidden text>`B333
`b
>>> Checkboxes
In addition to text fields, Checkboxes are another way of submitting data. They allow the user to make a single selection or select multiple options.
`Faaa
`=
`<?|field_name|value`>`b Label Text`
`=
When the checkbox is checked, it's field will be set to the provided value. If there are multiple checkboxes that share the same field name, the checked values will be concatenated when they are sent to the node by a comma.
``
`B444`<?|sign_up|1`>`b Sign me up`
You can also pre-check both checkboxes and radio groups by appending a |* after the field value.
`B444`<?|checkbox|1|*`>`b Pre-checked checkbox`
>>> Radio groups
Radio groups are another input that lets the user chose from a set of options. Unlike checkboxes, radio buttons with the same field name are mutually exclusive.
Example:
`=
`B900`<^|color|Red`>`b Red
`B090`<^|color|Green`>`b Green
`B009`<^|color|Blue`>`b Blue
`=
will render:
`B900`<^|color|Red`>`b Red
`B090`<^|color|Green`>`b Green
`B009`<^|color|Blue`>`b Blue
In this example, when the data is submitted, `B444` field_color`b will be set to whichever value from the list was selected.
`` ``

View File

@@ -160,7 +160,6 @@ def parse_line(line, state, url_delegate):
tw.in_columns = True tw.in_columns = True
else: else:
tw = urwid.Text(o, align=state["align"]) tw = urwid.Text(o, align=state["align"])
widgets.append((urwid.PACK, tw)) widgets.append((urwid.PACK, tw))
else: else:
if o["type"] == "field": if o["type"] == "field":
@@ -173,6 +172,36 @@ def parse_line(line, state, url_delegate):
f.field_name = fn f.field_name = fn
fa = urwid.AttrMap(f, fs) fa = urwid.AttrMap(f, fs)
widgets.append((fw, fa)) widgets.append((fw, fa))
elif o["type"] == "checkbox":
fn = o["name"]
fv = o["value"]
flabel = o["label"]
fs = o["style"]
fprechecked = o.get("prechecked", False)
f = urwid.CheckBox(flabel, state=fprechecked)
f.field_name = fn
f.field_value = fv
fa = urwid.AttrMap(f, fs)
widgets.append((urwid.PACK, fa))
elif o["type"] == "radio":
fn = o["name"]
fv = o["value"]
flabel = o["label"]
fs = o["style"]
fprechecked = o.get("prechecked", False)
if "radio_groups" not in state:
state["radio_groups"] = {}
if fn not in state["radio_groups"]:
state["radio_groups"][fn] = []
group = state["radio_groups"][fn]
f = urwid.RadioButton(group, flabel, state=fprechecked, user_data=fv)
f.field_name = fn
f.field_value = fv
fa = urwid.AttrMap(f, fs)
widgets.append((urwid.PACK, fa))
columns_widget = urwid.Columns(widgets, dividechars=0) columns_widget = urwid.Columns(widgets, dividechars=0)
text_widget = columns_widget text_widget = columns_widget
@@ -458,54 +487,100 @@ def make_output(state, line, url_delegate):
elif c == "a": elif c == "a":
state["align"] = state["default_align"] state["align"] = state["default_align"]
elif c == "<": elif c == '<':
if len(part) > 0:
output.append(make_part(state, part))
part = ""
try: try:
field_name = None field_start = i + 1 # position after '<'
field_name_end = line[i:].find("`") backtick_pos = line.find('`', field_start)
if field_name_end == -1: if backtick_pos == -1:
pass pass # No '`', invalid field
else: else:
field_name = line[i+1:i+field_name_end] field_content = line[field_start:backtick_pos]
field_name_skip = len(field_name)
field_masked = False field_masked = False
field_width = 24 field_width = 24
field_type = "field"
field_name = field_content
field_value = ""
field_data = ""
field_prechecked = False
if "|" in field_name: # check if field_content contains '|'
f_components = field_name.split("|") if '|' in field_content:
f_components = field_content.split('|')
field_flags = f_components[0] field_flags = f_components[0]
field_name = f_components[1] field_name = f_components[1]
if "!" in field_flags:
# handle field type indicators
if '^' in field_flags:
field_type = "radio"
field_flags = field_flags.replace("^", "")
elif '?' in field_flags:
field_type = "checkbox"
field_flags = field_flags.replace("?", "")
elif '!' in field_flags:
field_flags = field_flags.replace("!", "") field_flags = field_flags.replace("!", "")
field_masked = True field_masked = True
if len(field_flags) > 0:
field_width = min(int(field_flags), 256)
def sr(): # Handle field width
return "@{"+str(random.randint(1000,9999))+"}" if len(field_flags) > 0:
rsg = sr() try:
while rsg in line[i+field_name_end:]: field_width = min(int(field_flags), 256)
rsg = sr() except ValueError:
lr = line[i+field_name_end:].replace("\\>", rsg) pass # Ignore invalid width
endpos = lr.find(">")
# Check for value and pre-checked flag
if endpos == -1: if len(f_components) > 2:
pass field_value = f_components[2]
else:
field_value = ""
if len(f_components) > 3:
if f_components[3] == '*':
field_prechecked = True
else: else:
field_data = lr[1:endpos].replace(rsg, "\\>") # No '|', so field_name is field_content
skip = len(field_data)+field_name_skip+2 field_name = field_content
field_data = field_data.replace("\\>", ">") field_type = "field"
output.append({ field_masked = False
"type":"field", field_width = 24
"name": field_name, field_value = ""
"width": field_width, field_prechecked = False
"masked": field_masked,
"data": field_data, # Find the closing '>' character
"style": make_style(state) field_end = line.find('>', backtick_pos)
}) if field_end == -1:
pass # No closing '>', invalid field
else:
field_data = line[backtick_pos+1:field_end]
# Now, we have all field data
if field_type in ["checkbox", "radio"]:
# for checkboxes and radios, field_data is the label
output.append({
"type": field_type,
"name": field_name,
"value": field_value if field_value else field_data,
"label": field_data,
"prechecked": field_prechecked,
"style": make_style(state)
})
else:
# For text fields field_data is the initial text
output.append({
"type": "field",
"name": field_name,
"width": field_width,
"masked": field_masked,
"data": field_data,
"style": make_style(state)
})
skip = field_end - i
except Exception as e: except Exception as e:
pass pass
elif c == "[": elif c == "[":
endpos = line[i:].find("]") endpos = line[i:].find("]")
if endpos == -1: if endpos == -1:

View File

@@ -30,6 +30,6 @@ setuptools.setup(
entry_points= { entry_points= {
'console_scripts': ['nomadnet=nomadnet.nomadnet:main'] 'console_scripts': ['nomadnet=nomadnet.nomadnet:main']
}, },
install_requires=["rns>=0.7.7", "lxmf>=0.5.0", "urwid>=2.4.4", "qrcode"], install_requires=["rns>=0.8.8", "lxmf>=0.5.8", "urwid>=2.4.4", "qrcode"],
python_requires=">=3.6", python_requires=">=3.6",
) )