Skip to content

Commit

Permalink
refactoring for easier testing
Browse files Browse the repository at this point in the history
  • Loading branch information
MarkhamLee committed Mar 7, 2024
1 parent 59f8d40 commit b5b7089
Show file tree
Hide file tree
Showing 2 changed files with 259 additions and 0 deletions.
138 changes: 138 additions & 0 deletions etl_pipelines_nodejs/openweather_current_ts/src/main.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,138 @@
"use strict";
// (C) Markham Lee 2023 - 2024
// productivity-music-stocks-weather-IoT-dashboard
// https://github.com/MarkhamLee/productivity-music-stocks-weather-IoT-dashboard
// Node variant for the OpenWeather API ETL - pulls down data for current weather
// conditions and writes it to InfluxDB
var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
return new (P || (P = Promise))(function (resolve, reject) {
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
step((generator = generator.apply(thisArg, _arguments || [])).next());
});
};
var __generator = (this && this.__generator) || function (thisArg, body) {
var _ = { label: 0, sent: function() { if (t[0] & 1) throw t[1]; return t[1]; }, trys: [], ops: [] }, f, y, t, g;
return g = { next: verb(0), "throw": verb(1), "return": verb(2) }, typeof Symbol === "function" && (g[Symbol.iterator] = function() { return this; }), g;
function verb(n) { return function (v) { return step([n, v]); }; }
function step(op) {
if (f) throw new TypeError("Generator is already executing.");
while (g && (g = 0, op[0] && (_ = 0)), _) try {
if (f = 1, y && (t = op[0] & 2 ? y["return"] : op[0] ? y["throw"] || ((t = y["return"]) && t.call(y), 0) : y.next) && !(t = t.call(y, op[1])).done) return t;
if (y = 0, t) op = [op[0] & 2, t.value];
switch (op[0]) {
case 0: case 1: t = op; break;
case 4: _.label++; return { value: op[1], done: false };
case 5: _.label++; y = op[1]; op = [0]; continue;
case 7: op = _.ops.pop(); _.trys.pop(); continue;
default:
if (!(t = _.trys, t = t.length > 0 && t[t.length - 1]) && (op[0] === 6 || op[0] === 2)) { _ = 0; continue; }
if (op[0] === 3 && (!t || (op[1] > t[0] && op[1] < t[3]))) { _.label = op[1]; break; }
if (op[0] === 6 && _.label < t[1]) { _.label = t[1]; t = op; break; }
if (t && _.label < t[2]) { _.label = t[2]; _.ops.push(op); break; }
if (t[2]) _.ops.pop();
_.trys.pop(); continue;
}
op = body.call(thisArg, _);
} catch (e) { op = [6, e]; y = 0; } finally { f = t = 0; }
if (op[0] & 5) throw op[1]; return { value: op[0] ? op[1] : void 0, done: true };
}
};
Object.defineProperty(exports, "__esModule", { value: true });
exports.writeData = exports.parseData = exports.getWeatherData = void 0;
var axios_1 = require("axios");
var influxdb_client_1 = require("@influxdata/influxdb-client");
var openweather_config_1 = require("../utils/openweather_config");
var openweather_library_1 = require("../utils/openweather_library");
// Get OpenWeather data
var getWeatherData = function (weatherUrl) { return __awaiter(void 0, void 0, void 0, function () {
var data, error_1, message, full_message;
return __generator(this, function (_a) {
switch (_a.label) {
case 0:
_a.trys.push([0, 2, , 3]);
return [4 /*yield*/, axios_1.default.get(weatherUrl)];
case 1:
data = (_a.sent()).data;
return [2 /*return*/, data];
case 2:
error_1 = _a.sent();
message = "OpenWeather API Pipeline Current Weather (Nodejs variant) failure, API connection error: ";
full_message = message.concat(error_1.message);
(0, openweather_library_1.sendSlackAlerts)(full_message);
console.error(full_message);
return [2 /*return*/, {
message: error_1.message,
status: error_1.response.status,
}];
case 3: return [2 /*return*/];
}
});
}); };
exports.getWeatherData = getWeatherData;
// parse out the desired fields
// TODO: update to calculate AQI - may need all the fields for that
var parseData = function (data) {
// split out the part of the json that contains the bulk of the data points
var weather_data = data.main;
// parse out individual fields
var payload = { "barometric_pressure": weather_data.pressure,
"description": data.weather[0].description,
"feels_like": weather_data.feels_like,
"high": weather_data.temp_max,
"humidity": weather_data.humidity,
"low": weather_data.temp_min,
"temp": weather_data.temp,
"time_stamp": data.dt,
"weather": data.weather[0].main,
"wind": data.wind.speed };
// Validate the payload before writing to InfluxDB.
var status = (0, openweather_library_1.validateJson)(payload);
if (status == 1) {
return process.exit();
}
console.log('DB payload ready: ', payload);
return payload;
};
exports.parseData = parseData;
//method to write data to InfluxDB
// the InfluxDB node.js library doesn't have a clean way of just
// pushing json data to the DB. So, the write methods will have to
// live in the primary ETL code for now.
var writeData = function (payload) {
try {
var writeClient_1 = (0, openweather_library_1.createInfluxClient)(openweather_config_1.config.bucket);
var point_1 = new influxdb_client_1.Point(openweather_config_1.config.measurement)
.tag("OpenWeatherAPI", "current_weather")
.floatField('temp', payload.temp)
.floatField('wind', payload.wind)
.floatField('barometric_pressure', payload.barometric_pressure)
.floatField('humidity', payload.humidity)
.floatField('low', payload.low)
.floatField('high', payload.high)
.floatField('feels_like', payload.feels_like)
.intField('time_stamp', payload.time_stamp)
.stringField('description', payload.description)
.stringField('weather', payload.weather);
// write data to InfluxDB
void setTimeout(function () {
writeClient_1.writePoint(point_1);
console.log("Weather data successfully written to InfluxDB");
}, 1000);
// flush client
void setTimeout(function () {
// flush InfluxDB client
writeClient_1.flush();
}, 1000);
}
catch (error) {
var message = "OpenWeather API Pipeline Current Weather (Nodejs variant) failure, InfluxDB write error: ";
var full_message = (message.concat(JSON.stringify((error.body))));
console.error(full_message);
//send pipeline failure alert via Slack
(0, openweather_library_1.sendSlackAlerts)(full_message);
}
};
exports.writeData = writeData;
121 changes: 121 additions & 0 deletions etl_pipelines_nodejs/openweather_current_ts/src/main.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,121 @@
// (C) Markham Lee 2023 - 2024
// productivity-music-stocks-weather-IoT-dashboard
// https://github.com/MarkhamLee/productivity-music-stocks-weather-IoT-dashboard
// Node variant for the OpenWeather API ETL - pulls down data for current weather
// conditions and writes it to InfluxDB

import axios from 'axios';
import { Point } from '@influxdata/influxdb-client';
import {config, WeatherResponse, ErrorMessage} from "../utils/openweather_config";
import { createInfluxClient, sendSlackAlerts, validateJson, }
from "../utils/openweather_library";


// Get OpenWeather data
const getWeatherData = async (weatherUrl: string): Promise<WeatherResponse[] | ErrorMessage> => {

try {

const { data } = await axios.get<WeatherResponse[]>(weatherUrl)
return data

} catch (error: any){

const message = "OpenWeather API Pipeline Current Weather (Nodejs variant) failure, API connection error: "
const full_message = message.concat(error.message)
sendSlackAlerts(full_message)
console.error(full_message)

return {
message: error.message,
status: error.response.status,
}
}
}

// parse out the desired fields
// TODO: update to calculate AQI - may need all the fields for that
const parseData = (data: any) => {

// split out the part of the json that contains the bulk of the data points
const weather_data = data.main;

// parse out individual fields
const payload = {"barometric_pressure": weather_data.pressure,
"description": data.weather[0].description,
"feels_like": weather_data.feels_like,
"high": weather_data.temp_max,
"humidity": weather_data.humidity,
"low": weather_data.temp_min,
"temp": weather_data.temp,
"time_stamp": data.dt,
"weather": data.weather[0].main,
"wind": data.wind.speed }

// Validate the payload before writing to InfluxDB.
const status = validateJson(payload)

if (status == 1) {

return process.exit()

}

console.log('DB payload ready: ', payload)

return payload

}

//method to write data to InfluxDB
// the InfluxDB node.js library doesn't have a clean way of just
// pushing json data to the DB. So, the write methods will have to
// live in the primary ETL code for now.
const writeData = (payload: any) => {

try {

const writeClient = createInfluxClient(config.bucket)

let point = new Point(config.measurement)
.tag("OpenWeatherAPI", "current_weather")
.floatField('temp', payload.temp)
.floatField('wind', payload.wind)
.floatField('barometric_pressure', payload.barometric_pressure)
.floatField('humidity', payload.humidity)
.floatField('low', payload.low)
.floatField('high', payload.high)
.floatField('feels_like', payload.feels_like)
.intField('time_stamp', payload.time_stamp)
.stringField('description', payload.description)
.stringField('weather', payload.weather)

// write data to InfluxDB
void setTimeout(() => {

writeClient.writePoint(point);
console.log("Weather data successfully written to InfluxDB")
}, 1000)


// flush client
void setTimeout(() => {

// flush InfluxDB client
writeClient.flush()
}, 1000)

} catch (error: any) {

const message = "OpenWeather API Pipeline Current Weather (Nodejs variant) failure, InfluxDB write error: "
const full_message = (message.concat(JSON.stringify((error.body))));
console.error(full_message);

//send pipeline failure alert via Slack
sendSlackAlerts(full_message);

}

}

export {getWeatherData, parseData, writeData}

0 comments on commit b5b7089

Please sign in to comment.