[Draft] [gpx] Add gpx import #5166
No reviewers
Labels
No labels
Accessibility
Accessibility
Address
Address
Android
Android
Android Auto
Android Auto
Android Automotive (AAOS)
Android Automotive (AAOS)
API
API
AppGallery
AppGallery
AppStore
AppStore
Battery and Performance
Battery and Performance
Blocker
Blocker
Bookmarks and Tracks
Bookmarks and Tracks
Borders
Borders
Bug
Bug
Build
Build
CarPlay
CarPlay
Classificator
Classificator
Community
Community
Core
Core
CrashReports
CrashReports
Cycling
Cycling
Desktop
Desktop
DevEx
DevEx
DevOps
DevOps
dev_sandbox
dev_sandbox
Directions
Directions
Documentation
Documentation
Downloader
Downloader
Drape
Drape
Driving
Driving
Duplicate
Duplicate
Editor
Editor
Elevation
Elevation
Enhancement
Enhancement
Epic
Epic
External Map Datasets
External Map Datasets
F-Droid
F-Droid
Fonts
Fonts
Frequently User Reported
Frequently User Reported
Fund
Fund
Generator
Generator
Good first issue
Good first issue
Google Play
Google Play
GPS
GPS
GSoC
GSoC
iCloud
iCloud
Icons
Icons
iOS
iOS
Legal
Legal
Linux Desktop
Linux Desktop
Linux packaging
Linux packaging
Linux Phone
Linux Phone
Mac OS
Mac OS
Map Data
Map Data
Metro
Metro
Navigation
Navigation
Need Feedback
Need Feedback
Night Mode
Night Mode
NLnet 2024-06-281
NLnet 2024-06-281
No Feature Parity
No Feature Parity
Opening Hours
Opening Hours
Outdoors
Outdoors
POI Info
POI Info
Privacy
Privacy
Public Transport
Public Transport
Raw Idea
Raw Idea
Refactoring
Refactoring
Regional
Regional
Regression
Regression
Releases
Releases
RoboTest
RoboTest
Route Planning
Route Planning
Routing
Routing
Ruler
Ruler
Search
Search
Security
Security
Styles
Styles
Tests
Tests
Track Recording
Track Recording
Translations
Translations
TTS
TTS
UI
UI
UX
UX
Walk Navigation
Walk Navigation
Watches
Watches
Web
Web
Wikipedia
Wikipedia
Windows
Windows
Won't fix
Won't fix
World Map
World Map
No milestone
No project
No assignees
1 participant
Due date
No due date set.
Dependencies
No dependencies set.
Reference: organicmaps/organicmaps-tmp#5166
Loading…
Add table
Reference in a new issue
No description provided.
Delete branch "gpx"
Deleting a branch is permanent. Although the deleted branch may continue to exist for a short time before it actually gets removed, it CANNOT be undone in most cases. Continue?
Hi,
I try to finish the work started by @dvdmrtnz in #4067.
I made some tests and updated Manifest file but still was not able to have .gpx file associated with OrganicMaps. I asked on stackoverflow and searched on web, but was not able to find an answer.
@rtsisyk @arnaudvergnet - as Android experts could you give me any advice on it?
@biodranik - could you please advice, what minimal set of items is required to be implemented for merge? I checked your comment here #4067 so I think it should be enough to fix codestyle issues that you mentioned and add more tests (what do we want to check - large files, some details of GPX spec, something else)?
Closes #4067
Fixes #2953
Fixes #2681
Fixes #624
Thanks for your efforts, it's a good start! Let's focus on properly testing and supporting different GPX files/versions and 1.1 spec, and clean the code a bit.
@ -1108,3 +1091,3 @@
{
const vector<string> exts = { kKmzExtension, kKmlExtension, kKmbExtension };
const vector<string> exts = { kKmzExtension, kKmlExtension, kKmbExtension, kGpxExtension };
return jni::ToJavaStringArray(env, exts);
nit: use
std::array<std::string_view>
here, and add appropriate overload to jni::ToJavaStringArrayPlease find different examples of GPX files from different popular apps (some should be already in our issues), and make tests for them.
Also, please make a list of non-standard features that may be used in some GPX files (custom extensions, for example), that it would be useful to support, if not now, but in the future.
Can we support GPX 1.0 and GPX 2.0? What elements are not supported now from the full spec? It would be great to list/track them in the related issue. What elements are supported by other apps?
Can we use std::string_view instead? In any case, using namespace should be moved to the internal gpx namespace.
We don't use {} for one-liners, it makes code easier to read.
Using "wpt"sv here and in other cases could make comparisons/parsing faster.
Can GetTagFromEnd also return std::string_view?
Do you really need these tags? What does linter say?
Is it a blind copy-paste? Are all these variables needed/used?
Imported GPX should be converted/stored as KML. Is this change really needed?
@ -1875,0 +1877,4 @@
kmlData = LoadKmlFile(fileSavePath, KmlFileType::Text);
else if (ext == ".gpx")
kmlData = LoadKmlFile(fileSavePath, KmlFileType::Gpx);
else if (ext == ".kmb")
What are the benefits of storing files internally in different formats? Why is it a bad idea to convert imported GPX files to KML ones and store them internally in KML?
Yes, at least they are populated in the parser, I'll review their purpose.
Fixed
Fixed
Fixed
Fixed
Fixed
@ -1875,0 +1877,4 @@
kmlData = LoadKmlFile(fileSavePath, KmlFileType::Text);
else if (ext == ".gpx")
kmlData = LoadKmlFile(fileSavePath, KmlFileType::Gpx);
else if (ext == ".kmb")
I don't think that different formats are used internally, this parameter is used just to select a valid parser:
Fixed
@biodranik just to note that there's only version 1.0 and 1.1 of GPX.
Correct, my mistake.
I added several real-life tracks. I added very basic test to verify that files are parsed and we load track with waypoints.
Looks like files are more or less similar, except extensions in couple of them.
For extensions, I collected some of them on this page: https://github.com/cyber-toad/organicmaps/wiki/GPX. Not sure we have a place in UI to display them, except, maybe, color?
I added initial list here: https://github.com/cyber-toad/organicmaps/wiki/GPX
Looks like we don't have polygons, but for tracks they are not required.
Maybe load everything from metadata and put to track description?
Thanks!
There is no need to do many tests for similar/same tags.
Was there a list of some GPX extensions/examples already on our Github?
If you collect them in your fork, please link it in the corresponding issue.
Only those extensions that can be easily mapped to our current KML should be supported (colors?), everything else can stay in the TODO list.
P.S. there are some unresolved comments.
There is an issue about polygons for KML, AFAIR. It would be great to mention/link there GPX polygons TODO too.
I added links to comments with requests here: https://github.com/cyber-toad/organicmaps/wiki/GPX
Will have a look at colors implementation.
I saw unresolved comments, need some time for c++ changes...
Polygons added to https://github.com/cyber-toad/organicmaps/wiki/GPX
Looks like copy-paste from KML where they are used for some specific case and not used in GPX for now, so removed.
@ -1108,3 +1091,3 @@
{
const vector<string> exts = { kKmzExtension, kKmlExtension, kKmbExtension };
const vector<string> exts = { kKmzExtension, kKmlExtension, kKmbExtension, kGpxExtension };
return jni::ToJavaStringArray(env, exts);
Upd: looks like ToJavaStringArray should already support std::arraystd::string_view
Switched to string_view
Unused variables are removed.
Can you please clarify the current behavior? Are imported GPX always converted and stored internally as KML files? Then there's no need to patch kml load code.
I checked in debugger and looks like extension is checked on early stage of file loader where we still have .gpx:
@ -1108,3 +1091,3 @@
{
const vector<string> exts = { kKmzExtension, kKmlExtension, kKmbExtension };
const vector<string> exts = { kKmzExtension, kKmlExtension, kKmbExtension, kGpxExtension };
return jni::ToJavaStringArray(env, exts);
I replaced with
const std::array<std::string_view, 4> exts = { kKmzExtension, kKmlExtension, kKmbExtension, kGpxExtension };
but got an error:
jni_helper.hpp:89:32: error: no matching function for call to object of type '(lambda at /home/a/dev/organicmaps/android/jni/./app/organicmaps/core/jni_helper.hpp:108:22)'
so it is not supported for now?
@ -1108,3 +1091,3 @@
{
const vector<string> exts = { kKmzExtension, kKmlExtension, kKmbExtension };
const vector<string> exts = { kKmzExtension, kKmlExtension, kKmbExtension, kGpxExtension };
return jni::ToJavaStringArray(env, exts);
Did you made kKmzExtension and others also std::string_view ?
Am I correctly understood, that this part of the code is used during the import too?
Yes, it is executed when I import a track. Here is a stacktrace for this call.
@ -1108,3 +1091,3 @@
{
const vector<string> exts = { kKmzExtension, kKmlExtension, kKmbExtension };
const vector<string> exts = { kKmzExtension, kKmlExtension, kKmbExtension, kGpxExtension };
return jni::ToJavaStringArray(env, exts);
Thanks, it fixes the issue, but after this change I noticed that kKmlExtension is quite widely used in the code as std::string,
for example
std::string GenerateUniqueFileName(const std::string & path, std::string name, std::string const & ext = kKmlExtension);
auto const filePath = base::JoinPath(GetPlatform().TmpDir(), fileName + kKmlExtension);
etc. Do we want to rework all this code in scope of gpx change?
@ -1108,3 +1091,3 @@
{
const vector<string> exts = { kKmzExtension, kKmlExtension, kKmbExtension };
const vector<string> exts = { kKmzExtension, kKmlExtension, kKmbExtension, kGpxExtension };
return jni::ToJavaStringArray(env, exts);
You can make a separate PR for it, if it's an easy-to-do change :) Or add a commit here. As I raised this issue, I'm ok with that approach ("fix some old code if you're already touching it anyway").
@ -0,0 +126,4 @@
{
ASSERT_LESS(n, m_tags.size(), ());
return m_tags[m_tags.size() - n - 1];
}
auto const colorBytes?
LOG(LWARNING, ("Invalid color value", value));
Octal system is not understandable by many. Why not use just a plain 255 number?
@ -0,0 +177,4 @@
m_data.m_bookmarksData.push_back(std::move(data));
}
else if (GEOMETRY_TYPE_LINE == m_geometryType)
What is this color?
We don't use braces for one-line ifs and loops.
@ -0,0 +126,4 @@
{
ASSERT_LESS(n, m_tags.size(), ());
return m_tags[m_tags.size() - n - 1];
}
Fixed
@ -0,0 +126,4 @@
{
ASSERT_LESS(n, m_tags.size(), ());
return m_tags[m_tags.size() - n - 1];
}
Fixed
@ -0,0 +126,4 @@
{
ASSERT_LESS(n, m_tags.size(), ());
return m_tags[m_tags.size() - n - 1];
}
Fixed
Fixed
@ -0,0 +177,4 @@
m_data.m_bookmarksData.push_back(std::move(data));
}
else if (GEOMETRY_TYPE_LINE == m_geometryType)
Yes, was not able to find it's usage, removed.
Thank you very much! We're almost there :)
Can you please add three more layers of directories to gpx, kml and kmz? It could be an issue in your device if Android stores files "too deep" in many subfolders.
@ -1108,3 +1091,3 @@
{
const vector<string> exts = { kKmzExtension, kKmlExtension, kKmbExtension };
const vector<string> exts = { kKmzExtension, kKmlExtension, kKmbExtension, kGpxExtension };
return jni::ToJavaStringArray(env, exts);
Can you do this minor refactoring in a parallel PR?
@ -0,0 +3,4 @@
<metadata>
<name>new</name>
<author>
<name>gpx.studio</name>
Can you please test tracks recorded with GPSLogger: https://play.google.com/store/apps/details?id=eu.basicairdata.graziano.gpslogger ?
I tried it and there were no any name displayed on the recorded track. Was it a bug or the name was missing? If name was missing, the following logic can be implemented:
@ -11,0 +15,4 @@
<string>GPS Exchange Format (GPX)</string>
<key>LSHandlerRank</key>
<string>Default</string>
<key>LSItemContentTypes</key>
This entry doesn't look right or complete. Did you check https://github.com/osmandapp/OsmAnd-iOS/blob/master/OsmAnd%20Maps%20copy-Info.plist ?
@ -1,9 +1,10 @@
project(kml_tests)
set(SRC
Let's sort names alphabetically.
This is not our coding standard. It is described in docs. Please review the code and fix it everywhere.
Please mark all const variables with a
const
keyword in all places.ditto
constexpr here and below
std::string_view objects are faster when passed by value than by reference.
Does it fit in 120 characters?
@ -0,0 +70,4 @@
if (mercator::ValidX(m_org.x) && mercator::ValidY(m_org.y))
{
// Set default name.
if (m_name.empty())
Should name be set even if there is no m_org point? Is it a possible scenario?
@ -0,0 +95,4 @@
if (GetTagFromEnd(0) == gpx::kWpt)
m_geometryType = GEOMETRY_TYPE_POINT;
else if (GetTagFromEnd(0) == gpx::kTrkPt || GetTagFromEnd(0) == gpx::kRtePt)
m_geometryType = GEOMETRY_TYPE_LINE;
What happens when both ifs above won't work? Is it possible with area-only gpx files, for example? Will the parser crash?
@ -0,0 +245,4 @@
DeserializerGpx::DeserializerGpx(FileData & fileData)
: m_fileData(fileData)
{
m_fileData = {};
Why do you need to reset it after setting in L247 above? Is it a mistake?
@ -0,0 +20,4 @@
{
namespace gpx
{
class GpxParser
Can this class be inside kml::gpx namespace?
@ -0,0 +85,4 @@
reader.ReadAsString(gpxText);
if (!gpxText.empty() && gpxText[0] == '<')
LOG(LWARNING, (gpxText));
MYTHROW(DeserializeException, ("Could not parse GPX."));
Did you test if this exception is thrown correctly with a corrupted GPX?
Note the most popular extension comes first.
Fixed
Yes, fixed
Fixed
Fixed
Fixed
@ -1,9 +1,10 @@
project(kml_tests)
set(SRC
Fixed
Fixed
@ -11,0 +15,4 @@
<string>GPS Exchange Format (GPX)</string>
<key>LSHandlerRank</key>
<string>Default</string>
<key>LSItemContentTypes</key>
Fixed
@ -0,0 +3,4 @@
<metadata>
<name>new</name>
<author>
<name>gpx.studio</name>
By some reason I can't install this app on my android phone - it is displayed as not available in Google Play, could you please share the recorded file here (all points could be removed for privacy, I'm just interested in metadata and track data).
Also I didn't get how do you want to use file name, I think it is not available in parser? Also during the load there is some artificial name: organicmaps/organicmaps#5166 ?
@ -0,0 +3,4 @@
<metadata>
<name>new</name>
<author>
<name>gpx.studio</name>
Looks like it was a file from another logger: https://gpslogger.app/
It has only points in the log, without any metadata, with a file name like
20230402.gpx
:@ -0,0 +3,4 @@
<metadata>
<name>new</name>
<author>
<name>gpx.studio</name>
Ok, I see.
So we want to add filename as a parameter to
std::unique_ptr<kml::FileData> LoadKmlData(Reader const & reader, KmlFileType fileType)
method so it can pass it to GPX deserializer and it will fallback to it in case of missing name in file?
@ -0,0 +3,4 @@
<metadata>
<name>new</name>
<author>
<name>gpx.studio</name>
That can be an option, right.
Alternatively, is it better/easier to check if returned FileData has empty Category name, and fix it there?
@ -0,0 +3,4 @@
<metadata>
<name>new</name>
<author>
<name>gpx.studio</name>
Yes, thanks, it looks easier. I applied this change and added a test for this case.
Fixed
Fixed
Switched to string_view
Fixed
Fixed
@ -0,0 +95,4 @@
if (GetTagFromEnd(0) == gpx::kWpt)
m_geometryType = GEOMETRY_TYPE_POINT;
else if (GetTagFromEnd(0) == gpx::kTrkPt || GetTagFromEnd(0) == gpx::kRtePt)
m_geometryType = GEOMETRY_TYPE_LINE;
I added a test - UNIT_TEST(Empty). Parser is not crashed, but if there are no points in track - it is not extracted.
Thanks! There are also some unresolved comments from the previous review. Can you please take a look at them here? organicmaps/organicmaps#5166/files#
We plan to release a beta soon, it would be great if these changes will be ready to test on a wider audience ;-)
Indentation is wrong.
Can FileReader::ReadAsString() be used here?
Fixed
Fixed
Fixed
Fixed
Fixed
Fixed
Fixed
Fixed
Added consts
Fixed
Fixed
Fixed
Fixed
Fixed
@ -1108,3 +1091,3 @@
{
const vector<string> exts = { kKmzExtension, kKmlExtension, kKmbExtension };
const vector<string> exts = { kKmzExtension, kKmlExtension, kKmbExtension, kGpxExtension };
return jni::ToJavaStringArray(env, exts);
ok, let's cover it separately once mandatory functionality is implemented
@ -0,0 +20,4 @@
{
namespace gpx
{
class GpxParser
Fixed
Fixed
This line fixes the broken loading of kmz files.
thanks for suggestion, fixed
@ -0,0 +85,4 @@
reader.ReadAsString(gpxText);
if (!gpxText.empty() && gpxText[0] == '<')
LOG(LWARNING, (gpxText));
MYTHROW(DeserializeException, ("Could not parse GPX."));
I have a difficulty with testing it on Debug or with unit-test - it logs with error level if parser fails
coding/parse_xml.hpp:80
so both debug version on Android and unit test fails with "Abort. Log level is too serious" - what is a valid way to test it?
@ -0,0 +245,4 @@
DeserializerGpx::DeserializerGpx(FileData & fileData)
: m_fileData(fileData)
{
m_fileData = {};
I think that it was copy-pasted from KML, where it was added here:
b1ff412810 (diff-c16f904482bd8eb25901b3bee393895ac9744cd2495235cd1a2f6ecd509f8951R869)
I tried to remove it for gpx & kml and kml_tests passed, but I don't have a full understanding what happens with memory there - we just re-initialize memory passed to us to put deserializer result there?
@ -0,0 +70,4 @@
if (mercator::ValidX(m_org.x) && mercator::ValidY(m_org.y))
{
// Set default name.
if (m_name.empty())
The corrected question:
Should a similar empty name check be done for non-points (tracks) too, below in the GEOMETRY_TYPE_LINE if? How and when the default track name is generated if it's empty?
@ -0,0 +85,4 @@
reader.ReadAsString(gpxText);
if (!gpxText.empty() && gpxText[0] == '<')
LOG(LWARNING, (gpxText));
MYTHROW(DeserializeException, ("Could not parse GPX."));
@ -0,0 +85,4 @@
reader.ReadAsString(gpxText);
if (!gpxText.empty() && gpxText[0] == '<')
LOG(LWARNING, (gpxText));
MYTHROW(DeserializeException, ("Could not parse GPX."));
Use the necessary exception type if you expect it.
@ -0,0 +245,4 @@
DeserializerGpx::DeserializerGpx(FileData & fileData)
: m_fileData(fileData)
{
m_fileData = {};
False alarm. This code correctly clears the output structure before writing to it.
Please add an exception test in a follow-up pull request. We'll prepare these changes for the release.