-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathapp.js
157 lines (141 loc) · 5.36 KB
/
app.js
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
156
157
import express from 'express';
import axios from 'axios';
import dotenv from 'dotenv';
import { fileURLToPath } from 'url';
import { join, dirname } from 'path';
// Configuration
const __filename = fileURLToPath(import.meta.url);
const __dirname = dirname(__filename);
dotenv.config({ path: join(__dirname, 'config.env') });
// Constants
const TOMORROW_API_CONFIG = {
BASE_URL: process.env.BASE_URL,
API_KEY: process.env.TOMORROW_API_KEY,
LOCATION: {
LAT: '21.422510',
LON: '39.826168'
}
};
const PORT = process.env.PORT || 4000;
// Utility functions
const formatters = {
windSpeed: (speedInMS) => (speedInMS * 3.6).toFixed(1),
percentage: (value) => parseFloat(value).toFixed(1),
time: (isoString) => {
if (!isoString) return null;
try {
const date = new Date(isoString);
if (isNaN(date.getTime())) return null;
return date.toLocaleTimeString('en-US', {
hour: '2-digit',
minute: '2-digit',
hour12: true
});
} catch (error) {
console.error('Error formatting time:', error);
return null;
}
}
};
// API client
const weatherClient = {
headers: {
'Accept-Encoding': 'gzip, deflate, br',
'Accept': 'application/json'
},
async getCurrentWeather() {
const response = await axios.get(TOMORROW_API_CONFIG.BASE_URL, {
params: {
location: `${TOMORROW_API_CONFIG.LOCATION.LAT},${TOMORROW_API_CONFIG.LOCATION.LON}`,
apikey: TOMORROW_API_CONFIG.API_KEY,
units: 'metric',
timesteps: 'daily', // Changed from minutely to daily to get sunrise/sunset
fields: ['temperature', 'windSpeed', 'humidity', 'precipitationProbability', 'sunriseTime', 'sunsetTime']
},
headers: this.headers
});
const currentData = response.data.timelines.daily[0]?.values;
if (!currentData) {
throw new Error('Current weather data not found in response');
}
// Get current conditions from minutely data
const currentMinutely = await axios.get(TOMORROW_API_CONFIG.BASE_URL, {
params: {
location: `${TOMORROW_API_CONFIG.LOCATION.LAT},${TOMORROW_API_CONFIG.LOCATION.LON}`,
apikey: TOMORROW_API_CONFIG.API_KEY,
units: 'metric',
timesteps: 'minutely',
fields: ['temperature', 'windSpeed', 'humidity', 'precipitationProbability']
},
headers: this.headers
});
const minutelyData = currentMinutely.data.timelines.minutely[0]?.values;
return {
temperature: minutelyData.temperature,
windSpeed: formatters.windSpeed(minutelyData.windSpeed),
humidity: formatters.percentage(minutelyData.humidity),
rainChance: formatters.percentage(minutelyData.precipitationProbability),
sunrise: formatters.time(currentData.sunriseTime),
sunset: formatters.time(currentData.sunsetTime)
};
},
async getWeatherForecast() {
const response = await axios.get(TOMORROW_API_CONFIG.BASE_URL, {
params: {
location: `${TOMORROW_API_CONFIG.LOCATION.LAT},${TOMORROW_API_CONFIG.LOCATION.LON}`,
apikey: TOMORROW_API_CONFIG.API_KEY,
units: 'metric',
timesteps: '1d',
startTime: 'now',
endTime: 'nowPlus7d',
fields: ['temperature', 'weatherCode', 'sunriseTime', 'sunsetTime']
},
headers: this.headers
});
return response.data.timelines.daily.map(day => {
const date = new Date(day.time);
return {
dayName: date.toLocaleDateString('en-US', { weekday: 'long' }),
date: date.toLocaleDateString(),
temperature: {
avg: day.values.temperatureAvg,
max: day.values.temperatureMax,
min: day.values.temperatureMin
},
weatherCode: day.values.weatherCodeMax,
sunrise: formatters.time(day.values.sunriseTime),
sunset: formatters.time(day.values.sunsetTime),
};
});
}
};
// Express app setup
const app = express();
app.use(express.json());
// Routes
app.get('/', async (req, res) => {
try {
const [current, dailyTemperatures] = await Promise.all([
weatherClient.getCurrentWeather(),
weatherClient.getWeatherForecast()
]);
res.status(200).json({
location: {
lat: TOMORROW_API_CONFIG.LOCATION.LAT,
lon: TOMORROW_API_CONFIG.LOCATION.LON
},
current,
dailyTemperatures
});
} catch (error) {
console.error('Weather API Error:', error.response?.data || error.message);
res.status(500).json({
error: 'Failed to fetch weather data',
details: error.response?.data || error.message
});
}
});
// Start server
app.listen(PORT, () => {
console.log(`Server is listening on port ${PORT}!`);
});