Opened 2 months ago
Closed 5 weeks ago
#24014 closed defect (fixed)
JOSM removes "newline" characters \n from inscription when editing nodes
Reported by: | scyyy | Owned by: | taylor.smock |
---|---|---|---|
Priority: | normal | Milestone: | 24.11 |
Component: | Core | Version: | |
Keywords: | Cc: | gaben, simonpoole |
Description (last modified by )
I was adding operator:wikidata
tags to some nodes (e.g. https://www.openstreetmap.org/node/12344145506). These nodes have an inscription=
tag created by iD that's using \n
(literal newline characters, U+000A
, displayed as \n
if you're using the textbox tag editor), in its value to signify line breaks. According to https://wiki.openstreetmap.org/wiki/Key:inscription, this is a valid usage.
However, after uploading the changes via JOSM (changeset https://www.openstreetmap.org/changeset/159245848), the inscriptions all lost their \n
linebreaks. JOSM seems to have replaced them with spaces. It should not do that, especially not in a totally silent way when my edit didn't even change the tag value at all. I'm glad that I saw this happening and was able to revert it; pretty sure that not everyone would.
Adding the status report manually because the "Report bug" button would send me to Debian's bug tracker, but I don't assume this is Debian-specific behavior. Let me know if you need more information, and thanks for looking into this.
Revision:19253 Is-Local-Build:false Build-Date:2024-11-06 04:21:43 Debian-Release:0.0.svn19253+dfsg-1~bpo12+1 Build-Name:Debian Identification: JOSM/1.5 (19253 Debian en) Linux Debian GNU/Linux 12 (bookworm) Memory Usage: 432 MB / 11920 MB (171 MB allocated, but free) Java version: 17.0.13+11-Debian-2deb12u1, Debian, OpenJDK 64-Bit Server VM Look and Feel: javax.swing.plaf.metal.MetalLookAndFeel Screen: :0.0 1920x1080x[Multi depth]@60Hz (scaling 1.00×1.00) :0.1 2560x1080x[Multi depth]@[Unknown refresh rate] (scaling 1.00×1.00) :0.2 1080x1920x[Multi depth]@[Unknown refresh rate] (scaling 1.00×1.00) Maximum Screen Size: 2560×1920 Best cursor sizes: 16×16→16×16, 32×32→32×32 Environment variable LANG: en_US.UTF-8 System property file.encoding: UTF-8 System property sun.jnu.encoding: UTF-8 Locale info: en_US Numbers with default locale: 1234567890 -> 1234567890 Desktop environment: sway Java package: openjdk-17-jre:amd64-17.0.13+11-2~deb12u1 Java ATK Wrapper package: libatk-wrapper-java:all-0.40.0-3 libcommons-compress-java: libcommons-compress-java:all-1.22-1 libcommons-logging-java: libcommons-logging-java:- fonts-noto: fonts-noto:all-20201225-1 VM arguments: [--module-path=/usr/share/openjfx/lib, --add-modules=java.scripting,java.sql,javafx.controls,javafx.media,javafx.swing,javafx.web, -Djosm.restart=true, -Djava.net.useSystemProxies=true, -XX:MaxRAMPercentage=75.0, --add-exports=java.base/sun.security.action=ALL-UNNAMED, --add-exports=java.desktop/com.sun.imageio.plugins.jpeg=ALL-UNNAMED, --add-exports=java.desktop/com.sun.imageio.spi=ALL-UNNAMED] Plugins: + reverter (36343) Last errors/warnings: - 00015.681 W: java.lang.UnsupportedOperationException: The BROWSE action is not supported on the current platform! cache.bing.attribution.xml=1729713311 cache.capabilities1637351842=1731854021 cache.motd.html=1731854022 cache.motd.html.java=17.0.13 cache.motd.html.lang=En: cache.motd.html.version=19253 changesetdialog.lastHeight=0 commandstack.lastHeight=118 commandstack.visible=true conflict.lastHeight=0 download.tab=3 downloadprimitive.lasttype=2 draw.rawgps.lines=-1 filter.lastHeight=0 gui.geometry=x=0,y=0,width=1920,height=1061 imagery.entries=[{max-zoom=22, min-zoom=1, noTileHeaders={"X-VE-Tile-Info":["no-tile"]}, noTileChecksums={"MD5":["c13269481c73de6e18589f9fbc3bdf7e"]}, metadataHeaders={"X-VE-TILEMETA-CaptureDatesRange":"Capture Date"}, transparent=true, minimumTileExpire=3600, name=Bing aerial imagery, id=Bing, type=bing, url=https://www.bing.com/maps/, permission-reference-url=https://wiki.openstreetmap.org/wiki/Bing_Maps, cookies=, icon=..., customHttpHeaders={}, category=photo}, {max-zoom=22, noTileHeaders={"Etag":["\"10i954m13i2\""]}, noTileChecksums={"MD5":["f27d9de7f80c13501f470595e327aa6d"]}, transparent=true, minimumTileExpire=3600, name=Esri World Imagery, id=EsriWorldImagery, type=tms, url=https://{switch:services,server}.arcgisonline.com/arcgis/rest/services/World_Imagery/MapServer/tile/{zoom}/{y}/{x}, attribution-text=Terms & Feedback, attribution-url=https://wiki.openstreetmap.org/wiki/Esri, cookies=, icon=..., customHttpHeaders={}, category=photo}, {max-zoom=22, transparent=true, minimumTileExpire=3600, name=Esri World Imagery (Clarity) Beta, id=EsriWorldImageryClarity, type=tms, url=https://clarity.maptiles.arcgis.com/arcgis/rest/services/World_Imagery/MapServer/tile/{zoom}/{y}/{x}, attribution-text=Terms & Feedback, attribution-url=https://wiki.openstreetmap.org/wiki/Esri, permission-reference-url=https://github.com/osmlab/editor-layer-index/pull/358#issuecomment-361416110, cookies=, icon=..., description=Esri archive imagery that may be clearer and more accurate than the default layer., customHttpHeaders={}, category=photo}, {max-zoom=22, noTileHeaders={"ETag":["\"336b022ed883bc72347a637634e490d4\"|\"067736a547cafe90014b4e59b6510abe\"|\"ee1f6802b0234046b553cbbc672ac7d9\"|\"9f5a2f1d7cc131e58befc2052c71c827\""]}, noTileChecksums={"MD5":["58e78313d04adf0ea64b8de8590c3d46"]}, transparent=true, minimumTileExpire=3600, name=Mapbox Satellite, id=Mapbox, type=tms, url=https://{switch:a,b,c,d}.tiles.mapbox.com/v4/mapbox.satellite/{zoom}/{x}/{y}.jpg?access_token={apikey}, attribution-text=Terms & Feedback, attribution-url=https://www.mapbox.com/about/maps/, permission-reference-url=https://wiki.openstreetmap.org/wiki/Vertical_Aerial_Photographs#DigitalGlobe_.2F_MapBox, cookies=, icon=..., customHttpHeaders={"Referer":"https://josm.openstreetmap.de/"}, category=photo}, {max-zoom=19, valid-georeference=true, transparent=true, minimumTileExpire=3600, name=OpenStreetMap Carto (Standard), id=standard, type=tms, url=https://tile.openstreetmap.org/{zoom}/{x}/{y}.png, attribution-text=© OpenStreetMap contributors, attribution-url=https://www.openstreetmap.org/, permission-reference-url=https://wiki.osmfoundation.org/wiki/Terms_of_Use, cookies=, icon=..., customHttpHeaders={}, category=osmbasedmap} ] imagery.layers.default=[Bing, EsriWorldImagery, EsriWorldImageryClarity, Mapbox, standard] josm.version=19253 lastDirectory=${HOME}/Downloads layerlist.lastHeight=120 mappaint.renderer-class-name=org.openstreetmap.josm.data.osm.visitor.paint.StyledMapRenderer mappaint.style.known-defaults=[resource://styles/standard/elemstyles.mapcss] mapstyle.lastHeight=0 minimap.lastHeight=0 mirror.<josm.cache>/images.https://josm.openstreetmap.de/raw-attachment/wiki/Help/Josm_main.jpg.svg=[1729712735281, <josm.cache>/images/mirror_https___josm.openstreetmap.de_raw-attachment_wiki_Help_Josm_main.jpg.svg] mirror.https://josm.openstreetmap.de/maps=[1731854024228, <josm.cache>/mirror_https___josm.openstreetmap.de_maps] mirror.https://josm.openstreetmap.de/remote/geofabrik-index-v1-nogeom.json=[1731854023462, <josm.cache>/mirror_https___josm.openstreetmap.de_remote_geofabrik-index-v1-nogeom.json] mirror.https://josm.openstreetmap.de/rules=[1731855784970, <josm.cache>/mirror_https___josm.openstreetmap.de_rules] notes/note_open.lastHeight=0 org.openstreetmap.josm.actions.SessionSaveAction$SessionSaveAsDialog.geometry=x=735,y=305,width=450,height=450 org.openstreetmap.josm.gui.dialogs.properties.TagEditHelper$AddTagsDialog.geometry=x=786,y=334,width=347,height=392 org.openstreetmap.josm.gui.dialogs.properties.TagEditHelper$EditTagDialog.geometry=x=813,y=449,width=294,height=163 org.openstreetmap.josm.gui.dialogs.relation.GenericRelationEditor.geometry=x=1006,y=180,width=700,height=650 org.openstreetmap.josm.gui.download.DownloadDialog.geometry=x=460,y=230,width=1000,height=600 org.openstreetmap.josm.gui.download.DownloadObjectDialog.primitivesHistory=[17010855] org.openstreetmap.josm.gui.help.HelpBrowser.geometry=x=0,y=0,width=600,height=400 org.openstreetmap.josm.gui.history.HistoryBrowserDialogManager.geometry=x=535,y=280,width=850,height=500 org.openstreetmap.josm.gui.io.UploadDialog.geometry=x=560,y=230,width=800,height=600 org.openstreetmap.josm.gui.oauth.OAuthAuthorizationWizard.geometry=x=572,y=323,width=776,height=415 org.openstreetmap.josm.gui.preferences.PreferenceDialog.geometry=x=560,y=130,width=800,height=800 osm-download.bounds=51.8584713;10.0445938;51.9131439;10.1132584 osm-server.upload-strategy=singlerequest pluginmanager.lastupdate=1731855660945 pluginmanager.version=19253 plugins=[reverter] preferences.reset.draw.rawgps.lines=true properties.recent-tags=[power, terminal, man_made, pipeline, name, Fernwasserleitung Söse Nord, operator:wikidata, Q1588057, operator, Harzwasserwerke, substance, water, location, underground, layer, -1, fixme, continue] propertiesdialog.lastHeight=178 relationlist.lastHeight=178 remotecontrol.permission.authorization=true reverter.ChangesetIdQuery.changesetsHistory=[159245848] search.history=[R operator=Harzwasserwerke "operator:wikidata"=, R marker=post operator=Harzwasserwerke inscription=, R marker=post operator=Harzwasserwerke, R marker=post, R modified OR deleted] selectionlist.lastHeight=178 sensitive.keys=[oauth.access-token.key, oauth.access-token.object.OAuth20.api.openstreetmap.org, oauth.access-token.parameters.OAuth20.api.openstreetmap.org, oauth.access-token.secret, osm-server.password, osm-server.username, proxy.pass, proxy.user, sensitive.keys] toggleDialogs.width=326 upload.comment.history=[Add section of FWL Söse Nord between Dannhausen and Gremsheim to relation, Add Wikidata tag for Harzwasserwerke-operated items, Add, , Expand Dannhausen power line, modify some other small things along the way.] upload.comment.last-used=1731859817 upload.source.history=[survey, knowledge, , survey;aerial imagery, Bing] userlist.lastHeight=0 validator.lastHeight=179 validator.visible=true
Attachments (2)
Change History (20)
comment:1 by , 2 months ago
Milestone: | → 24.11 |
---|
comment:2 by , 2 months ago
comment:3 by , 2 months ago
Cc: | added |
---|
comment:4 by , 2 months ago
No, but it is breaking valid data, so it should be fixed. As far as I see iD does not use escape sequences, but only displays it this way. I don't think we support this at all. Return signs are probably stripped.
Tasks would be
- don't break data
- allow to enter such data
There is an UTF8 sign für newline. I'd prefer that over escape sequences in display.
comment:5 by , 2 months ago
Description: | modified (diff) |
---|
Just for clarification, since I found the wiki on inscription=
somewhat ambiguous, I've looked into whether iD is using \n
as two characters in the tag (backslash, then n
), or actual linebreaks (U+000A
, one byte), and it's the latter. https://github.com/openstreetmap/iD/issues/10326 confirms that.
I've updated the ticket description accordingly and removed my initial "I'm not sure whether it's one or two characters".
comment:6 by , 2 months ago
Owner: | changed from | to
---|---|
Status: | new → assigned |
@stoecker: I agree, it is breaking valid data, and it should be fixed. I just didn't want to redo work that you were doing (since you had assigned it to milestone:24.11).
comment:7 by , 2 months ago
Cc: | added |
---|
Quick comments on what happens:
- We attempt to clean up keys/values by trimming whitespace and replacing all whitespace with spaces. See source:trunk/src/org/openstreetmap/josm/tools/Utils.java#L822 for details.
Options to fix:
- Add new method somewhere (e.g. source:trunk/src/org/openstreetmap/josm/data/osm/OsmUtils.java)
- Example:
OsmUtils.trimIfAllowed(String key, String value)
and have a blacklist of keys that should not be trimmed or only trim if no preset has the key value as freeform text.
- Example:
- Don't replace newlines in values
- Don't trim whitespace in long values
I'm personally inclined to only trim if a no preset has the key value as freeform text.
cons:
name
-- this would introduce potential issues where people copy/paste a name, e.g.name= foobar
(notice spaces on both ends).
So, if we go the preset route, we should add an additional xml attribute to indicate that a value may have what would otherwise be extraneous whitespace.
@simonpoole: Do you already have an extension for that in vespucci presets? I didn't see anything in https://vespucci.io/tutorials/presets/#extensions , although value_type
is probably the closest extension you have (and probably the "easiest" one to extend, maybe with notrim
or something).
comment:8 by , 2 months ago
What about a "multiline" flag for such elements? And not totally stop stripping, but only allow <CR>?
comment:9 by , 2 months ago
multiline
is definitely an option as a flag (e.g. <item [...] multiline="true">
versus <item [...] value_type="multiline">
).
I'll give simonpoole a chance to respond since Vespucci uses the same preset format, and I don't want to introduce potential incompatibilities.
comment:10 by , 2 months ago
From discussion on IRC today, it looks like adding multiline=true|false
and normalize=true|false
will work for Vespucci. The defaults (for JOSM) will be multiline=false
and normalize=true
.
comment:12 by , 8 weeks ago
Actually, I am not sure if only using the presets works and it would tie some other GUI elements to the presets such as the Add/Change Value dialogs need to be aware and need to support multiline
.
What happens with keys which are not in the default presets?
By the way, a lot of other dialogs which display the value need at least to display the line break somehow (TagsMemberships Panel, History Dialog, Conflict Dialog …)
by , 8 weeks ago
Attachment: | 24014.multilinenormalize.patch added |
---|
XSD changes for multiline
and normalize
attributes
by , 8 weeks ago
Attachment: | 24014.valuetypenormalize.patch added |
---|
Add value_type
(with multiline
as an option) and normalize
attributes
follow-up: 15 comment:13 by , 8 weeks ago
I've attached two patches for the XSD, but (from comment:10), I'm going to be using attachment:24014.multilinenormalize.patch. attachment:24014.valuetypenormalize.patch is an alternative that I discussed with simonpoole.
@skyper:
Yes, some work would need to be done on frontend elements to properly display and edit multiline values. This isn't high priority though, since step one is "stop breaking current data".
As far as keys not in a preset, defaulting to "current" behavior is the best option, IMO. Realistically, inscription
is an oddball, and the only other keys (AFAIK) that should have "similar" behavior is fixme
and note
tags.
comment:15 by , 8 weeks ago
Replying to taylor.smock:
As far as keys not in a preset, defaulting to "current" behavior is the best option, IMO. Realistically,
inscription
is an oddball, and the only other keys (AFAIK) that should have "similar" behavior isfixme
andnote
tags.
description
is a candidate too.
comment:16 by , 7 weeks ago
Regarding fixme and note, description: About twenty years OSM could live without carriage return in these tags. I would not suggest to encourage usage now.
I consider carriage return in inscription a very bad idea as well, but it seems we have to live with it. But that doesn't mean it should be easy to produce such entries :-)
comment:18 by , 5 weeks ago
Milestone: | 24.12 → 24.11 |
---|---|
Resolution: | → fixed |
Status: | assigned → closed |
I'm going to call this fixed for now. If we need to do anything with other tags, those can have separate tickets opened.
@stoecker: are you working on this?