Modify

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 scyyy)

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)

24014.multilinenormalize.patch (1.3 KB ) - added by taylor.smock 8 weeks ago.
XSD changes for multiline and normalize attributes
24014.valuetypenormalize.patch (4.0 KB ) - added by taylor.smock 8 weeks ago.
Add value_type (with multiline as an option) and normalize attributes

Download all attachments as: .zip

Change History (20)

comment:1 by stoecker, 2 months ago

Milestone: 24.11

comment:2 by taylor.smock, 2 months ago

@stoecker: are you working on this?

comment:3 by gaben, 2 months ago

Cc: gaben added

comment:4 by stoecker, 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 scyyy, 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 taylor.smock, 2 months ago

Owner: changed from team to taylor.smock
Status: newassigned

@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 taylor.smock, 2 months ago

Cc: simonpoole added

Quick comments on what happens:

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.
  • 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 stoecker, 2 months ago

What about a "multiline" flag for such elements? And not totally stop stripping, but only allow <CR>?

comment:9 by taylor.smock, 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 taylor.smock, 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:11 by stoecker, 2 months ago

Sounds like a good solution.

comment:12 by skyper, 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 taylor.smock, 8 weeks ago

XSD changes for multiline and normalize attributes

by taylor.smock, 8 weeks ago

Add value_type (with multiline as an option) and normalize attributes

comment:13 by taylor.smock, 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:14 by taylor.smock, 8 weeks ago

In 19261/josm:

See #24014: Add multiline and normalize attributes to preset xsd

normalize="false" will prevent all whitespace normalization while
normalize="true" + multiline="true" will strip start and end whitespace and
inner whitespace that is not newlines.

The primary reason for this change is osmwiki:Key:inscription which can have
newlines in order to match the inscription.

This does not modify UI elements to support multiline editing.

in reply to:  13 comment:15 by SimonPoole, 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 is fixme and note tags.

description is a candidate too.

comment:16 by stoecker, 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:17 by taylor.smock, 7 weeks ago

Milestone: 24.1124.12

Ticket retargeted after milestone closed

comment:18 by taylor.smock, 5 weeks ago

Milestone: 24.1224.11
Resolution: fixed
Status: assignedclosed

I'm going to call this fixed for now. If we need to do anything with other tags, those can have separate tickets opened.

Modify Ticket

Change Properties
Set your email in Preferences
Action
as closed The owner will remain taylor.smock.
as The resolution will be set.
The resolution will be deleted. Next status will be 'reopened'.

Add Comment


E-mail address and name can be saved in the Preferences .
 
Note: See TracTickets for help on using tickets.