Google Earth NO2 data different from Sentinel 5 harp data

Hello,

I downloaded NO2 Sentinel 5P data for 6th and 7th of July 2020 for a specific location and performed harp ingest on it. Then I extracted Google Earth NO2 data for the same location and time. When I created a visualisation for both the products to make sure they are same, the visualisations look different.

The image on the left is the Google Earth NO2 data visualisation and the image on the right is the harp NO2 data visualisation:
image

My question is, if google earth data are already harp products then why do these visualisations look different? How can I verify the harp products generated from a netCDF file if they do not match the google earth data?

I would really appreciate any help!

Thanks!

This is because Google is applying the quality filtering as prescribed by the product readme file. You should do the same by applying a filtering on the tropospheric_NO2_column_number_density_validity variable (which is also what Google is doing). I would strongly suggest that you read the other threads on this forum. There are many that deal with reading S5P NO2 data with HARP.

I’ve tried the tropospheric_NO2_column_number_density_validity filtering but the visualisation looks way more different after applying this.

So I assumed may be cloud filtering was not done on the google earth data. Why do these visualisations look so different? I followed everything exactly like the readme file.

Thanks

PS - I passed ‘tropospheric_NO2_column_number_density_validity > 75’ into harp import product function operations.

Have you looked at Sentinel-5P OFFL NO2: Offline Nitrogen Dioxide? Google has documented quite well how they create their data products. They are using >50 as qa_value threshold (which means they also include cloud covered scenes).

I have looked at the google documentation and followed the exact same harp conversion but the visualisation still doesn’t match. This is what my operations parameter in harp import_product function looks like:

harp.import_product(‘S5P_OFFL_L2__NO2____20200707T104926_20200707T123056_14160_01_010302_20200709T034441.nc’,operations=‘tropospheric_NO2_column_number_density_validity > 50;bin_spatial(1975,38.025,0.01,2355,11.815,0.01);derive(longitude{longitude});derive(latitude{latitude});keep(NO2_column_number_density,tropospheric_NO2_column_number_density, stratospheric_NO2_column_number_density,NO2_slant_column_number_density, tropopause_pressure,absorbing_aerosol_index,cloud_fraction, sensor_altitude,sensor_azimuth_angle, sensor_zenith_angle,solar_azimuth_angle,solar_zenith_angle, latitude, longitude)’)"

and this is from the google document:

I am still not sure why are the visualisations different when I’m doing the exact same thing. On changing the validity filter to > 50 instead of > 75 I’m getting the same visualisation I got in the first graph I posted. So I still have the same issue.

Thanks

I checked myself, and it seems google is using the > 75 filter (so their documentation is off).

You seem to be only looking at 1 orbit, which is from 2020-07-07. But you need to merge everything from both 2020-07-06 and 2020-07-07.

If I combine the two main orbits S5P_OFFL_L2__NO2____20200706T110830_20200706T125000_14146_01_010302_20200708T040350.nc and S5P_OFFL_L2__NO2____20200707T104926_20200707T123056_14160_01_010302_20200709T034441.nc and use:

harp.import_product(['S5P_OFFL_L2__NO2____20200706T110830_20200706T125000_14146_01_010302_20200708T040350.nc', 'S5P_OFFL_L2__NO2____20200707T104926_20200707T123056_14160_01_010302_20200709T034441.nc'], operations='tropospheric_NO2_column_number_density_validity > 75;bin_spatial(1975,38.025,0.01,2355,11.815,0.01);derive(longitude{longitude});derive(latitude{latitude});keep(NO2_column_number_density,tropospheric_NO2_column_number_density, stratospheric_NO2_column_number_density,NO2_slant_column_number_density, tropopause_pressure,absorbing_aerosol_index,cloud_fraction, sensor_altitude,sensor_azimuth_angle, sensor_zenith_angle,solar_azimuth_angle,solar_zenith_angle, latitude, longitude)', reduction_operations="bin()")

I am getting something that is already pretty close to what you get from google earth engine.

1 Like

I tried it, I got an error that reduction_operations does not exist but when I removed it and ran it again, I got something pretty close to the actual data. Thank you! :slight_smile:

Sorry that was a mistake in the example. It should say post_operations="bin()". This is what merges the different orbits together.

1 Like

Perfect! Thank you :slight_smile: