Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Take the field index, not name, in the feature getters and setters #581

Merged
merged 1 commit into from
Nov 27, 2024

Conversation

lnicola
Copy link
Member

@lnicola lnicola commented Oct 29, 2024

  • I agree to follow the project's code of conduct.
  • I added an entry to CHANGES.md if knowledge of this change could be valuable to users.

Field access by name tends to be very slow (I've seen it take 30% of the time in some programs), but beginners are unlikely to realize that they're doing a number of expensive lookups per feature.

Summary of changes:

  • drop field getters that take a field name
  • change the field setters to take an index instead of a name (the index versions were missing)
  • make field_count return usize for consistency
  • add Defn::field_index and make Feature::field_index public
  • drop LayerAccess::create_feature_fields (it could be updated, but it's a bit weird)

@lnicola lnicola force-pushed the field-index branch 2 times, most recently from afbd4d9 to 27c7df4 Compare October 29, 2024 19:18
@lnicola
Copy link
Member Author

lnicola commented Oct 29, 2024

r? @metasim, @ChristianBeilschmidt, @jdroenner since this is a relatively big change
r? @latot since you said you wanted to get involved

Hopefully I'll be able to get rid of this thing soon:

image

(it goes on and on)

src/vector/defn.rs Show resolved Hide resolved
@latot
Copy link

latot commented Oct 29, 2024

I think this is nice, some questions... can we actually change the fields in any way that could broke the field id?

From where is the image? I can see some transformations from isize to i32 with as.

@lnicola
Copy link
Member Author

lnicola commented Oct 29, 2024

can we actually change the fields in any way that could broke the field id?

You can delete fields from the Defn of a layer, but it won't reflect in the fields of the features, and I don't think we even expose that.

From where is the image? I can see some transformations from isize to i32 with as.

From usize, maybe, but most of those are checked (I think layer iteration is the exception). GDAL fields are 32-bit and positive (-1 means a missing field), but Rust uses probably expect usize. We do a similar thing for raster dimensions. But it's a small change and can be reverted easily (either now or in the future).

Or we could go the other way around and introduce a FieldIdx type that wraps an i32.

@lnicola
Copy link
Member Author

lnicola commented Nov 13, 2024

Gentle nudge @jdroenner 😅.

@latot
Copy link

latot commented Nov 13, 2024

I thinking the next thing.

Actually you said there is no function exposed that would be able to change the columns (remove/add), at some point I think would be good to have some tests that checks the actual idx do not change in any of this operations, just to be sure this is going to work fine in the future too.

No idea if rn is the best moment.

@lnicola
Copy link
Member Author

lnicola commented Nov 13, 2024

@latot if we don't expose OGR_L_DeleteField, how could we test that it doesn't affect the other indexes? Note that the GDAL docs explicitly say:

This function should not be called while there are feature objects in existence that were obtained or created with the previous layer definition.

Keep in mind that the intended usage is something like:

let id_idx = layer.defn().field_index("id")?;
let name_idx = layer.defn().field_index("name")?;
// you could call `layer.defn().delete_field(name_idx)?` here if it existed, but why?

for feature in layer.features() {
    let id = feature.field_as_integer(id_idx)?.expect("id is set");
    let name = feature.field_as_string(id_idx)?.expect("name is set");
    println!("Feature {id}, name: {name}");

    // you could never call `layer.defn().delete_field(name_idx)?` here because Rust
}

@latot
Copy link

latot commented Nov 13, 2024

Yes, the would be the usage, is more like a note to when something that could affect the id is exposed to the user, also, the only side I'm seeing, some ppl, could try to call that function directly even if is not exposed, would be the last concern from me.

@lnicola
Copy link
Member Author

lnicola commented Nov 13, 2024

We can't prevent every bug. You can always pass in the wrong field index or name, or even delete the whole layer. But if you do, you'll notice pretty quickly once you run your program.

What's harder to notice is that your program is running twice as slow because you've used the wrong field access methods.

@lnicola lnicola mentioned this pull request Nov 23, 2024
2 tasks
@lnicola
Copy link
Member Author

lnicola commented Nov 27, 2024

Heads-up: I'll probably merge this tonight if there's no more opposition.

@lnicola lnicola merged commit 0ba909a into georust:master Nov 27, 2024
13 checks passed
@lnicola lnicola deleted the field-index branch November 27, 2024 16:00
@Atreyagaurav Atreyagaurav mentioned this pull request Nov 29, 2024
2 tasks
@lnicola lnicola mentioned this pull request Jan 10, 2025
2 tasks
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

3 participants