1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
|
import logging
import random
from datetime import datetime
from typing import Any, Dict, Optional
from urllib.parse import urlencode
from discord import Embed
from discord.ext.commands import Cog, Context, group
from bot.bot import SeasonalBot
from bot.constants import Tokens
logger = logging.getLogger(__name__)
NASA_BASE_URL = "https://api.nasa.gov"
NASA_IMAGES_BASE_URL = "https://images-api.nasa.gov"
NASA_EPIC_BASE_URL = "https://epic.gsfc.nasa.gov"
APOD_DEFAULT_PARAMS = {
"api_key": Tokens.nasa
}
class Space(Cog):
"""Space Cog contains commands, that show images, facts or other information about space."""
def __init__(self, bot: SeasonalBot):
self.bot = bot
self.http_session = bot.http_session
@group(name="space")
async def space(self, ctx: Context) -> None:
"""Head command that contains commands about space."""
await ctx.send_help("space")
@space.command(name="apod")
async def apod(self, ctx: Context, date: Optional[str] = None) -> None:
"""Get Astronomy Picture of Day from NASA API. Date is optional parameter, what formatting is YYYY-MM-DD."""
# Make copy of parameters
params = APOD_DEFAULT_PARAMS.copy()
# Parse date to params, when provided. Show error message when invalid formatting
if date:
try:
params["date"] = datetime.strptime(date, "%Y-%m-%d").date().isoformat()
except ValueError:
await ctx.send(f"Invalid date {date}. Please make sure your date is in format YYYY-MM-DD.")
return
# Do request to NASA API
result = await self.fetch_from_nasa("planetary/apod", params)
# Create embed from result
embed = Embed(title=f"Astronomy Picture of the Day - {result['date']}", description=result["explanation"])
embed.set_image(url=result["url"])
embed.set_footer(text="Powered by NASA API")
await ctx.send(embed=embed)
@space.command(name="nasa")
async def nasa(self, ctx: Context) -> None:
"""Get random NASA information/facts + image."""
page = random.randint(1, 50)
# Create params for request, create URL and do request
params = {
"media_type": "image",
"page": page
}
async with self.http_session.get(url=f"{NASA_IMAGES_BASE_URL}/search?{urlencode(params)}") as resp:
data = await resp.json()
# Get (random) item from result, that will be shown
item = random.choice(data["collection"]["items"])
# Create embed and send it
embed = Embed(title=item["data"][0]["title"], description=item["data"][0]["description"])
embed.set_image(url=item["links"][0]["href"])
embed.set_footer(text="Powered by NASA API")
await ctx.send(embed=embed)
@space.command(name="earth")
async def earth(self, ctx: Context) -> None:
"""Get one of latest random image of earth from NASA API."""
# Generate URL and make request to API
async with self.http_session.get(url=f"{NASA_EPIC_BASE_URL}/api/natural") as resp:
data = await resp.json()
# Get random item from result that will be shown
item = random.choice(data)
# Split date for image URL
year, month, day = item["date"].split(" ")[0].split("-")
image_url = f"{NASA_EPIC_BASE_URL}/archive/natural/{year}/{month}/{day}/jpg/{item['image']}.jpg"
# Create embed, fill and send it
embed = Embed(title="Earth Image", description=item["caption"])
embed.set_image(url=image_url)
embed.set_footer(text=f"Identifier: {item['identifier']} \u2022 Powered by NASA API")
await ctx.send(embed=embed)
@space.command(name="mars")
async def mars(self, ctx: Context, date: str) -> None:
"""
Get random Mars image by date.
Date formatting is YYYY-MM-DD. Current max date is 2019-09-28 and min 2012-08-06.
"""
# Create API request parameters, try to parse date
params = {
"api_key": Tokens.nasa
}
try:
params["earth_date"] = datetime.strptime(date, "%Y-%m-%d")
except ValueError:
await ctx.send(f"Invalid date {date}. Please make sure your date is in format YYYY-MM-DD.")
return
result = await self.fetch_from_nasa("mars-photos/api/v1/rovers/curiosity/photos", params)
# Check for empty result
if len(result["photos"]) < 1:
err_msg = (
f"We can't find result in date {date}. "
"**Note:** Current dates must in range 2012-08-06 and 2019-09-28."
)
await ctx.send(err_msg)
return
# Get random item from result, generate embed with it and send
item = random.choice(result["photos"])
embed = Embed(title=f"{item['rover']['name']}'s {item['camera']['full_name']} Mars Image")
embed.set_image(url=item["img_src"])
embed.set_footer(text="Powered by NASA API")
await ctx.send(embed=embed)
async def fetch_from_nasa(self, endpoint: str, params: Dict[str, Any]) -> Dict[str, Any]:
"""Fetch information from NASA API, return result."""
# Generate request URL from base URL, endpoint and parsed params
async with self.http_session.get(url=f"{NASA_BASE_URL}/{endpoint}?{urlencode(params)}") as resp:
return await resp.json()
def setup(bot: SeasonalBot) -> None:
"""Load Space Cog."""
# Check does bot have NASA API key in .env, when not, don't load Cog and print warning
if not Tokens.nasa:
logger.warning("Can't find NASA API key. Not loading Space Cog.")
return
bot.add_cog(Space(bot))
|