blob: 84a130aefdd9c977cd773365fd75bb1e39077b31 [file] [log] [blame]
/****************************************************************
* Licensed to the Apache Software Foundation (ASF) under one *
* or more contributor license agreements. See the NOTICE file *
* distributed with this work for additional information *
* regarding copyright ownership. The ASF licenses this file *
* to you 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 org.apache.james.jspf.policies;
import org.apache.james.jspf.core.DNSLookupContinuation;
import org.apache.james.jspf.core.DNSRequest;
import org.apache.james.jspf.core.DNSResponse;
import org.apache.james.jspf.core.SPF1Constants;
import org.apache.james.jspf.core.SPF1Record;
import org.apache.james.jspf.core.SPF1Utils;
import org.apache.james.jspf.core.SPFChecker;
import org.apache.james.jspf.core.SPFCheckerDNSResponseListener;
import org.apache.james.jspf.core.SPFSession;
import org.apache.james.jspf.core.exceptions.NeutralException;
import org.apache.james.jspf.core.exceptions.NoneException;
import org.apache.james.jspf.core.exceptions.PermErrorException;
import org.apache.james.jspf.core.exceptions.TempErrorException;
import org.apache.james.jspf.core.exceptions.TimeoutException;
import java.util.Iterator;
import java.util.List;
/**
* Get the raw dns txt or spf entry which contains a spf entry
*/
public class SPFRetriever implements SPFChecker {
private static final class SPFRecordHandlerDNSResponseListener implements SPFCheckerDNSResponseListener {
/**
* @see org.apache.james.jspf.core.SPFCheckerDNSResponseListener#onDNSResponse(org.apache.james.jspf.core.DNSResponse, org.apache.james.jspf.core.SPFSession)
*/
public DNSLookupContinuation onDNSResponse(
DNSResponse response, SPFSession session)
throws PermErrorException,
NoneException, TempErrorException,
NeutralException {
List<String> spfR;
try {
spfR = response.getResponse();
String record = extractSPFRecord(spfR);
if (record != null) {
session.setAttribute(SPF1Utils.ATTRIBUTE_SPF1_RECORD, new SPF1Record(record));
}
} catch (TimeoutException e) {
throw new TempErrorException("Timeout querying dns");
}
return null;
}
}
private static final class SPFRetrieverDNSResponseListener implements SPFCheckerDNSResponseListener {
/**
* @see org.apache.james.jspf.core.SPFCheckerDNSResponseListener#onDNSResponse(org.apache.james.jspf.core.DNSResponse, org.apache.james.jspf.core.SPFSession)
*/
public DNSLookupContinuation onDNSResponse(
DNSResponse response, SPFSession session)
throws PermErrorException, NoneException,
TempErrorException, NeutralException {
try {
List<String> spfR = response.getResponse();
if (spfR == null || spfR.isEmpty()) {
String currentDomain = session.getCurrentDomain();
return new DNSLookupContinuation(new DNSRequest(currentDomain, DNSRequest.TXT), new SPFRecordHandlerDNSResponseListener());
} else {
String record = extractSPFRecord(spfR);
if (record != null) {
session.setAttribute(SPF1Utils.ATTRIBUTE_SPF1_RECORD, new SPF1Record(record));
}
}
return null;
} catch (TimeoutException e) {
throw new TempErrorException("Timeout querying dns");
}
}
}
/**
* This is used for testing purpose. Setting this to true will skip the initial
* lookups for SPF records and instead will simply check the TXT records.
*/
private static final boolean CHECK_ONLY_TXT_RECORDS = false;
/**
* Return the extracted SPF-Record
*
* @param spfR the List which holds TXT/SPF - Records
* @return returnValue the extracted SPF-Record
* @throws PermErrorException if more then one SPF - Record was found in the
* given List.
*/
protected static String extractSPFRecord(List<String> spfR) throws PermErrorException {
if (spfR == null || spfR.isEmpty()) return null;
String returnValue = null;
Iterator<String> all = spfR.iterator();
while (all.hasNext()) {
// DO NOT trim the result!
String compare = all.next();
// We trim the compare value only for the comparison
if (compare.toLowerCase().trim().startsWith(SPF1Constants.SPF_VERSION1 + " ") || compare.trim().equalsIgnoreCase(SPF1Constants.SPF_VERSION1)) {
if (returnValue == null) {
returnValue = compare;
} else {
throw new PermErrorException(
"More than 1 SPF record found");
}
}
}
return returnValue;
}
/**
* @see org.apache.james.jspf.core.SPFChecker#checkSPF(org.apache.james.jspf.core.SPFSession)
*/
public DNSLookupContinuation checkSPF(SPFSession spfData)
throws PermErrorException, TempErrorException, NeutralException,
NoneException {
SPF1Record res = (SPF1Record) spfData.getAttribute(SPF1Utils.ATTRIBUTE_SPF1_RECORD);
if (res == null) {
String currentDomain = spfData.getCurrentDomain();
if (CHECK_ONLY_TXT_RECORDS) {
return new DNSLookupContinuation(new DNSRequest(currentDomain, DNSRequest.TXT), new SPFRecordHandlerDNSResponseListener());
} else {
return new DNSLookupContinuation(new DNSRequest(currentDomain, DNSRequest.SPF), new SPFRetrieverDNSResponseListener());
}
}
return null;
}
}