/*
* Copyright 2012 LinkedIn Corp.
*
* Licensed under the Apache License, Version 2.0 (the "License"); you may not
* use this file except in compliance with the License. You may obtain a copy of
* the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
* License for the specific language governing permissions and limitations under
* the License.
*/
package azkaban.trigger.builtin;
import java.util.HashMap;
import java.util.Map;
import org.apache.log4j.Logger;
import azkaban.alert.Alerter;
import azkaban.executor.ExecutableFlow;
import azkaban.sla.SlaOption;
import azkaban.trigger.TriggerAction;
import azkaban.ServiceProvider;
import azkaban.executor.AlerterHolder;
import azkaban.executor.ExecutorLoader;
public class SlaAlertAction implements TriggerAction {
public static final String type = "AlertAction";
private static final Logger logger = Logger.getLogger(SlaAlertAction.class);
private final String actionId;
private final SlaOption slaOption;
private final int execId;
private final AlerterHolder alerters;
private final ExecutorLoader executorLoader;
//todo chengren311: move this class to executor module when all existing triggers in db are expired
public SlaAlertAction(String id, SlaOption slaOption, int execId) {
this.actionId = id;
this.slaOption = slaOption;
this.execId = execId;
this.alerters = ServiceProvider.SERVICE_PROVIDER.getInstance(AlerterHolder.class);
this.executorLoader = ServiceProvider.SERVICE_PROVIDER.getInstance(ExecutorLoader.class);
}
@Override
public String getId() {
return actionId;
}
@Override
public String getType() {
return type;
}
@SuppressWarnings("unchecked")
public static SlaAlertAction createFromJson(Object obj) throws Exception {
return createFromJson((HashMap<String, Object>) obj);
}
public static SlaAlertAction createFromJson(HashMap<String, Object> obj)
throws Exception {
Map<String, Object> jsonObj = (HashMap<String, Object>) obj;
if (!jsonObj.get("type").equals(type)) {
throw new Exception("Cannot create action of " + type + " from "
+ jsonObj.get("type"));
}
String actionId = (String) jsonObj.get("actionId");
SlaOption slaOption = SlaOption.fromObject(jsonObj.get("slaOption"));
int execId = Integer.valueOf((String) jsonObj.get("execId"));
return new SlaAlertAction(actionId, slaOption, execId);
}
@Override
public TriggerAction fromJson(Object obj) throws Exception {
return createFromJson(obj);
}
@Override
public Object toJson() {
Map<String, Object> jsonObj = new HashMap<String, Object>();
jsonObj.put("actionId", actionId);
jsonObj.put("type", type);
jsonObj.put("slaOption", slaOption.toObject());
jsonObj.put("execId", String.valueOf(execId));
return jsonObj;
}
@Override
public void doAction() throws Exception {
logger.info("Alerting on sla failure.");
Map<String, Object> alert = slaOption.getInfo();
if (alert.containsKey(SlaOption.ALERT_TYPE)) {
String alertType = (String) alert.get(SlaOption.ALERT_TYPE);
Alerter alerter = alerters.get(alertType);
if (alerter != null) {
try {
ExecutableFlow flow = executorLoader.fetchExecutableFlow(execId);
alerter.alertOnSla(slaOption, SlaOption.createSlaMessage(slaOption, flow));
} catch (Exception e) {
e.printStackTrace();
logger.error("Failed to alert by " + alertType);
}
} else {
logger.error("Alerter type " + alertType
+ " doesn't exist. Failed to alert.");
}
}
}
@Override
public void setContext(Map<String, Object> context) {
}
@Override
public String getDescription() {
return type + " for " + execId + " with " + slaOption.toString();
}
}