Chapter 5¶
This is the fifth in a series of notebooks related to astronomy data.
As a continuing example, we will replicate part of the analysis in a recent paper, “Off the beaten path: Gaia reveals GD-1 stars outside of the main stream” by Adrian M. Price-Whelan and Ana Bonaca.
Picking up where we left off, the next step in the analysis is to select candidate stars based on photometry. The following figure from the paper is a color-magnitude diagram for the stars selected based on proper motion:
In red is a theoretical isochrone, showing where we expect the stars in GD-1 to fall based on the metallicity and age of their original globular cluster.
By selecting stars in the shaded area, we can further distinguish the main sequence of GD-1 from younger background stars.
Outline¶
Here are the steps in this notebook:
We’ll reload the candidate stars we identified in the previous notebook.
Then we’ll run a query on the Gaia server that uploads the table of candidates and uses a
JOINoperation to select photometry data for the candidate stars.We’ll write the results to a file for use in the next notebook.
After completing this lesson, you should be able to
Upload a table to the Gaia server.
Write ADQL queries involving
JOINoperations.
Installing libraries¶
If you are running this notebook on Colab, you can run the following cell to install Astroquery and a the other libraries we’ll use.
If you are running this notebook on your own computer, you might have to install these libraries yourself.
If you are using this notebook as part of a Carpentries workshop, you should have received setup instructions.
TODO: Add a link to the instructions.
# If we're running on Colab, install libraries
import sys
IN_COLAB = 'google.colab' in sys.modules
if IN_COLAB:
!pip install astroquery astro-gala pyia python-wget
Reloading the data¶
The following cell downloads the data from the previous notebook.
import os
from wget import download
filename = 'gd1_candidates.hdf5'
path = 'https://github.com/AllenDowney/AstronomicalData/raw/main/data/'
if not os.path.exists(filename):
print(download(path+filename))
And we can read it back.
import pandas as pd
candidate_df = pd.read_hdf(filename, 'candidate_df')
candidate_df is the Pandas DataFrame that contains results from the query in the previous notebook, which selects stars likely to be in GD-1 based on proper motion. It also includes position and proper motion transformed to the ICRS frame.
import matplotlib.pyplot as plt
x = candidate_df['phi1']
y = candidate_df['phi2']
plt.plot(x, y, 'ko', markersize=0.3, alpha=0.3)
plt.xlabel('ra (degree GD1)')
plt.ylabel('dec (degree GD1)');
This is the same figure we saw at the end of the previous notebook. GD-1 is visible against the background stars, but we will be able to see it more clearly after selecting based on photometry data.
Getting photometry data¶
The Gaia dataset contains some photometry data, including the variable bp_rp, which we used in the original query to select stars with BP - RP color between -0.75 and 2.
Selecting stars with bp-rp less than 2 excludes many class M dwarf stars, which are low temperature, low luminosity. A star like that at GD-1’s distance would be hard to detect, so if it is detected, it it more likely to be in the foreground.
Now, to select stars with the age and metal richness we expect in GD-1, we will use g - i color and apparent g-band magnitude, which are available from the Pan-STARRS survey.
Conveniently, the Gaia server provides data from Pan-STARRS as a table in the same database we have been using, so we can access it by making ADQL queries.
In general, looking up a star from the Gaia catalog and finding the corresponding star in the Pan-STARRS catalog is not easy. This kind of cross matching is not always possible, because a star might appear in one catalog and not the other. And even when both stars are present, there might not be a clear one-to-one relationship between stars in the two catalogs.
Fortunately, smart people have worked on this problem, and the Gaia database includes cross-matching tables that suggest a best neighbor in the Pan-STARRS catalog for many stars in the Gaia catalog.
This document describes the cross matching process. Briefly, it uses a cone search to find possible matches in approximately the right position, then uses attributes like color and magnitude to choose pairs of stars most likely to be identical.
So the hard part of cross-matching has been done for us. However, using the results is a little tricky.
But, it is also an opportunity to learn about one of the most important tools for working with databases: “joining” tables.
In general, a “join” is an operation where you match up records from one table with records from another table using as a “key” a piece of information that is common to both tables, usually some kind of ID code.
In this example:
Stars in the Gaia dataset are identified by
source_id.Stars in the Pan-STARRS dataset are identified by
obj_id.
For each candidate star we have selected so far, we have the source_id; the goal is to find the obj_id for the same star (we hope) in the Pan-STARRS catalog.
To do that we will:
Make a table that contains the
source_idfor each candidate star and upload the table to the Gaia server;Use the
JOINoperator to look up eachsource_idin thegaiadr2.panstarrs1_best_neighbourtable, which contains theobj_idof the best match for each star in the Gaia catalog; thenUse the
JOINoperator again to look up eachobj_idin thepanstarrs1_original_validtable, which contains the Pan-STARRS photometry data we want.
Let’s start with the first step, uploading a table.
Preparing a table for uploading¶
For each candidate star, we want to find the corresponding row in the gaiadr2.panstarrs1_best_neighbour table.
In order to do that, we have to:
Write the table in a local file as an XML VOTable, which is a format suitable for transmitting a table over a network.
Write an ADQL query that refers to the uploaded table.
Change the way we submit the job so it uploads the table before running the query.
The first step is not too difficult because Astropy provides a function called writeto that can write a Table in XML.
The documentation of this process is here.
First we have to convert our Pandas DataFrame to an Astropy Table.
from astropy.table import Table
candidate_table = Table.from_pandas(candidate_df)
type(candidate_table)
astropy.table.table.Table
To write the file, we can use Table.write with format='votable', as described here.
table = candidate_table[['source_id']]
table.write('candidate_df.xml', format='votable', overwrite=True)
Notice that we select a single column from the table, source_id.
We could write the entire table to a file, but that would take longer to transmit over the network, and we really only need one column.
This process, taking a structure like a Table and translating it into a form that can be transmitted over a network, is called serialization.
XML is one of the most common serialization formats. One nice feature is that XML data is plain text, as opposed to binary digits, so you can read the file we just wrote:
!head candidate_df.xml
<?xml version="1.0" encoding="utf-8"?>
<!-- Produced with astropy.io.votable version 4.0.1.post1
http://www.astropy.org/ -->
<VOTABLE version="1.4" xmlns="http://www.ivoa.net/xml/VOTable/v1.4" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:noNamespaceSchemaLocation="http://www.ivoa.net/xml/VOTable/v1.4">
<RESOURCE type="results">
<TABLE>
<FIELD ID="source_id" datatype="long" name="source_id"/>
<DATA>
<TABLEDATA>
<TR>
XML is a general format, so different XML files contain different kinds of data. In order to read an XML file, it’s not enough to know that it’s XML; you also have to know the data format, which is called a schema.
In this example, the schema is VOTable; notice that one of the first tags in the file specifies the schema, and even includes the URL where you can get its definition.
So this is an example of a self-documenting format.
A drawback of XML is that it tends to be big, which is why we wrote just the source_id column rather than the whole table.
The size of the file is about 750 KB, so that’s not too bad.
!ls -lh candidate_df.xml
-rw-rw-r-- 1 downey downey 396K Oct 19 14:48 candidate_df.xml
If you are using Windows, ls might not work; in that case, try:
!dir candidate_df.xml
Exercise: There’s a gotcha here we want to warn you about. Why do you think we used double brackets to specify the column we wanted? What happens if you use single brackets?
Run these cells to find out.
table = candidate_table[['source_id']]
type(table)
astropy.table.table.Table
column = candidate_table['source_id']
type(column)
astropy.table.column.Column
# writeto(column, 'candidate_df.xml')
Uploading a table¶
The next step is to upload this table to the Gaia server and use it as part of a query.
Here’s the documentation that explains how to run a query with an uploaded table.
In the spirit of incremental development and testing, let’s start with the simplest possible query.
query = """SELECT *
FROM tap_upload.candidate_df
"""
This query downloads all rows and all columns from the uploaded table. The name of the table has two parts: tap_upload specifies a table that was uploaded using TAP+ (remember that’s the name of the protocol we’re using to talk to the Gaia server).
And candidate_df is the name of the table, which we get to choose (unlike tap_upload, which we didn’t get to choose).
Here’s how we run the query:
from astroquery.gaia import Gaia
job = Gaia.launch_job_async(query=query,
upload_resource='candidate_df.xml',
upload_table_name='candidate_df')
Created TAP+ (v1.2.1) - Connection:
Host: gea.esac.esa.int
Use HTTPS: True
Port: 443
SSL Port: 443
Created TAP+ (v1.2.1) - Connection:
Host: geadata.esac.esa.int
Use HTTPS: True
Port: 443
SSL Port: 443
INFO: Query finished. [astroquery.utils.tap.core]
upload_resource specifies the name of the file we want to upload, which is the file we just wrote.
upload_table_name is the name we assign to this table, which is the name we used in the query.
And here are the results:
results = job.get_results()
results
| source_id |
|---|
| int64 |
| 635559124339440000 |
| 635860218726658176 |
| 635674126383965568 |
| 635535454774983040 |
| 635497276810313600 |
| 635614168640132864 |
| 635821843194387840 |
| 635551706931167104 |
| 635518889086133376 |
| 635580294233854464 |
| ... |
| 612282738058264960 |
| 612485911486166656 |
| 612386332668697600 |
| 612296172717818624 |
| 612250375480101760 |
| 612394926899159168 |
| 612288854091187712 |
| 612428870024913152 |
| 612256418500423168 |
| 612429144902815104 |
If things go according to plan, the result should contain the same rows and columns as the uploaded table.
len(candidate_table), len(results)
(7346, 7346)
set(candidate_table['source_id']) == set(results['source_id'])
True
In this example, we uploaded a table and then downloaded it again, so that’s not too useful.
But now that we can upload a table, we can join it with other tables on the Gaia server.
Joining with an uploaded table¶
Here’s the first example of a query that contains a JOIN clause.
query1 = """SELECT *
FROM gaiadr2.panstarrs1_best_neighbour as best
JOIN tap_upload.candidate_df as candidate_df
ON best.source_id = candidate_df.source_id
"""
Let’s break that down one clause at a time:
SELECT *means we will download all columns from both tables.FROM gaiadr2.panstarrs1_best_neighbour as bestmeans that we’ll get the columns from the Pan-STARRS best neighbor table, which we’ll refer to using the short namebest.JOIN tap_upload.candidate_df as candidate_dfmeans that we’ll also get columns from the uploaded table, which we’ll refer to using the short namecandidate_df.ON best.source_id = candidate_df.source_idspecifies that we will usesource_idto match up the rows from the two tables.
Here’s the documentation of the best neighbor table.
Let’s run the query:
job1 = Gaia.launch_job_async(query=query1,
upload_resource='candidate_df.xml',
upload_table_name='candidate_df')
INFO: Query finished. [astroquery.utils.tap.core]
And get the results.
results1 = job1.get_results()
results1
| source_id | original_ext_source_id | angular_distance | number_of_neighbours | number_of_mates | best_neighbour_multiplicity | gaia_astrometric_params | source_id_2 |
|---|---|---|---|---|---|---|---|
| arcsec | |||||||
| int64 | int64 | float64 | int32 | int16 | int16 | int16 | int64 |
| 635860218726658176 | 130911385187671349 | 0.053667035895467084 | 1 | 0 | 1 | 5 | 635860218726658176 |
| 635674126383965568 | 130831388428488720 | 0.038810268141577516 | 1 | 0 | 1 | 5 | 635674126383965568 |
| 635535454774983040 | 130631378377657369 | 0.034323028828991076 | 1 | 0 | 1 | 5 | 635535454774983040 |
| 635497276810313600 | 130811380445631930 | 0.04720255413250006 | 1 | 0 | 1 | 5 | 635497276810313600 |
| 635614168640132864 | 130571395922140135 | 0.020304189709964143 | 1 | 0 | 1 | 5 | 635614168640132864 |
| 635598607974369792 | 130341392091279513 | 0.036524626853403054 | 1 | 0 | 1 | 5 | 635598607974369792 |
| 635737661835496576 | 131001399333502136 | 0.036626827820716606 | 1 | 0 | 1 | 5 | 635737661835496576 |
| 635850945892748672 | 132011398654934147 | 0.021178742393378396 | 1 | 0 | 1 | 5 | 635850945892748672 |
| 635600532119713664 | 130421392285893623 | 0.04518820915043015 | 1 | 0 | 1 | 5 | 635600532119713664 |
| ... | ... | ... | ... | ... | ... | ... | ... |
| 612241781249124608 | 129751343755995561 | 0.04235715830001815 | 1 | 0 | 1 | 5 | 612241781249124608 |
| 612332147361443072 | 130141341458538777 | 0.02265249859012977 | 1 | 0 | 1 | 5 | 612332147361443072 |
| 612426744016802432 | 130521346852465656 | 0.03247653009961843 | 1 | 0 | 1 | 5 | 612426744016802432 |
| 612331739340341760 | 130111341217793839 | 0.036064240818025735 | 1 | 0 | 1 | 5 | 612331739340341760 |
| 612282738058264960 | 129741340445933519 | 0.025293237353496898 | 1 | 0 | 1 | 5 | 612282738058264960 |
| 612386332668697600 | 130351354570219774 | 0.02010316001403086 | 1 | 0 | 1 | 5 | 612386332668697600 |
| 612296172717818624 | 129691338006168780 | 0.051264212025836205 | 1 | 0 | 1 | 5 | 612296172717818624 |
| 612250375480101760 | 129741346475897464 | 0.031783740347530905 | 1 | 0 | 1 | 5 | 612250375480101760 |
| 612394926899159168 | 130581355199751795 | 0.04019174830546698 | 1 | 0 | 1 | 5 | 612394926899159168 |
| 612256418500423168 | 129931349075297310 | 0.009242789669513156 | 1 | 0 | 1 | 5 | 612256418500423168 |
This table contains all of the columns from the best neighbor table, plus the single column from the uploaded table.
results1.colnames
['source_id',
'original_ext_source_id',
'angular_distance',
'number_of_neighbours',
'number_of_mates',
'best_neighbour_multiplicity',
'gaia_astrometric_params',
'source_id_2']
Because one of the column names appears in both tables, the second instance of source_id has been appended with the suffix _2.
The length of the results table is about 2000, which means we were not able to find matches for all stars in the list of candidate_df.
len(results1)
3724
To get more information about the matching process, we can inspect best_neighbour_multiplicity, which indicates for each star in Gaia how many stars in Pan-STARRS are equally likely matches.
For this kind of data exploration, we’ll convert a column from the table to a Pandas Series so we can use value_counts, which counts the number of times each value appears in a Series, like a histogram.
import pandas as pd
nn = pd.Series(results1['best_neighbour_multiplicity'])
nn.value_counts()
1 3724
dtype: int64
The result shows that 1 is the only value in the Series, appearing xxx times.
That means that in every case where a match was found, the matching algorithm identified a single neighbor as the most likely match.
Similarly, number_of_mates indicates the number of other stars in Gaia that match with the same star in Pan-STARRS.
nm = pd.Series(results1['number_of_mates'])
nm.value_counts()
0 3724
dtype: int64
For this set of candidate_df, almost all of the stars we’ve selected from Pan-STARRS are only matched with a single star in the Gaia catalog.
Detail The table also contains number_of_neighbors which is the number of stars in Pan-STARRS that match in terms of position, before using other critieria to choose the most likely match.
Getting the photometry data¶
The most important column in results1 is original_ext_source_id which is the obj_id we will use to look up the likely matches in Pan-STARRS to get photometry data.
The process is similar to what we just did to look up the matches. We will:
Make a table that contains
source_idandoriginal_ext_source_id.Write the table to an XML VOTable file.
Write a query that joins the uploaded table with
gaiadr2.panstarrs1_original_validand selects the photometry data we want.Run the query using the uploaded table.
Since we’ve done everything here before, we’ll do these steps as an exercise.
Exercise: Select source_id and original_ext_source_id from results1 and write the resulting table as a file named external.xml.
# Solution
table = results1[['source_id', 'original_ext_source_id']]
table.write('external.xml', format='votable', overwrite=True)
Use !head to confirm that the file exists and contains an XML VOTable.
!head external.xml
<?xml version="1.0" encoding="utf-8"?>
<!-- Produced with astropy.io.votable version 4.0.1.post1
http://www.astropy.org/ -->
<VOTABLE version="1.4" xmlns="http://www.ivoa.net/xml/VOTable/v1.4" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:noNamespaceSchemaLocation="http://www.ivoa.net/xml/VOTable/v1.4">
<RESOURCE type="results">
<TABLE>
<FIELD ID="source_id" datatype="long" name="source_id" ucd="meta.id;meta.main">
<DESCRIPTION>
Unique Gaia source identifier
</DESCRIPTION>
Exercise: Read the documentation of the Pan-STARRS table and make note of obj_id, which contains the object IDs we’ll use to find the rows we want.
Write a query that uses each value of original_ext_source_id from the uploaded table to find a row in gaiadr2.panstarrs1_original_valid with the same value in obj_id, and select all columns from both tables.
Suggestion: Develop and test your query incrementally. For example:
Write a query that downloads all columns from the uploaded table. Test to make sure we can read the uploaded table.
Write a query that downloads the first 10 rows from
gaiadr2.panstarrs1_original_valid. Test to make sure we can access Pan-STARRS data.Write a query that joins the two tables and selects all columns. Test that the join works as expected.
As a bonus exercise, write a query that joins the two tables and selects just the columns we need:
source_idfrom the uploaded tableg_mean_psf_magfromgaiadr2.panstarrs1_original_validi_mean_psf_magfromgaiadr2.panstarrs1_original_valid
Hint: When you select a column from a join, you have to specify which table the column is in.
# Solution
query2 = """SELECT *
FROM tap_upload.external as external
"""
# Solution
query2 = """SELECT TOP 10 *
FROM gaiadr2.panstarrs1_original_valid
"""
# Solution
query2 = """SELECT *
FROM gaiadr2.panstarrs1_original_valid as ps
JOIN tap_upload.external as external
ON ps.obj_id = external.original_ext_source_id
"""
# Solution
query2 = """SELECT
external.source_id, ps.g_mean_psf_mag, ps.i_mean_psf_mag
FROM gaiadr2.panstarrs1_original_valid as ps
JOIN tap_upload.external as external
ON ps.obj_id = external.original_ext_source_id
"""
print(query2)
SELECT
external.source_id, ps.g_mean_psf_mag, ps.i_mean_psf_mag
FROM gaiadr2.panstarrs1_original_valid as ps
JOIN tap_upload.external as external
ON ps.obj_id = external.original_ext_source_id
job2 = Gaia.launch_job_async(query=query2,
upload_resource='external.xml',
upload_table_name='external')
INFO: Query finished. [astroquery.utils.tap.core]
results2 = job2.get_results()
results2
| source_id | g_mean_psf_mag | i_mean_psf_mag |
|---|---|---|
| mag | ||
| int64 | float64 | float64 |
| 635860218726658176 | 17.8978004455566 | 17.5174007415771 |
| 635674126383965568 | 19.2873001098633 | 17.6781005859375 |
| 635535454774983040 | 16.9237995147705 | 16.478099822998 |
| 635497276810313600 | 19.9242000579834 | 18.3339996337891 |
| 635614168640132864 | 16.1515998840332 | 14.6662998199463 |
| 635598607974369792 | 16.5223999023438 | 16.1375007629395 |
| 635737661835496576 | 14.5032997131348 | 13.9849004745483 |
| 635850945892748672 | 16.5174999237061 | 16.0450000762939 |
| 635600532119713664 | 20.4505996704102 | 19.5177001953125 |
| ... | ... | ... |
| 612241781249124608 | 20.2343997955322 | 18.6518001556396 |
| 612332147361443072 | 21.3848991394043 | 20.3076000213623 |
| 612426744016802432 | 17.8281002044678 | 17.4281005859375 |
| 612331739340341760 | 21.8656997680664 | 19.5223007202148 |
| 612282738058264960 | 22.5151996612549 | 19.9743995666504 |
| 612386332668697600 | 19.3792991638184 | 17.9923000335693 |
| 612296172717818624 | 17.4944000244141 | 16.926700592041 |
| 612250375480101760 | 15.3330001831055 | 14.6280002593994 |
| 612394926899159168 | 16.4414005279541 | 15.8212003707886 |
| 612256418500423168 | 20.8715991973877 | 19.9612007141113 |
Challenge exercise
Do both joins in one query.
There’s an example here you could start with.
Write the data¶
Since we have the data in an Astropy Table, let’s store it in a FITS file.
filename = 'gd1_photo.fits'
results2.write(filename, overwrite=True)
We can check that the file exists, and see how big it is.
!ls -lh gd1_photo.fits
-rw-rw-r-- 1 downey downey 96K Oct 19 14:49 gd1_photo.fits
At around 175 KB, it is smaller than some of the other files we’ve been working with.
If you are using Windows, ls might not work; in that case, try:
!dir gd1_photo.fits
Summary¶
In this notebook, we used database JOIN operations to select photometry data for the stars we’ve identified as candidates to be in GD-1.
In the next notebook, we’ll use this data for a second round of selection, identifying stars that have photometry data consistent with GD-1.
Best practice¶
Use
JOINoperations to combine data from multiple tables in a databased, using some kind of identifier to match up records from one table with records from another.This is another example of a practice we saw in the previous notebook, moving the computation to the data.