/projects/connector-manager/source/java/com/google/enterprise/connector/servlet/RemoteAddressFilter.java

http://google-enterprise-connector-manager.googlecode.com/ · Java · 139 lines · 69 code · 11 blank · 59 comment · 11 complexity · fb2832f88a9ae42ff70b639c8c4f2a15 MD5 · raw file

  1. // Copyright 2010 Google Inc.
  2. //
  3. // Licensed under the Apache License, Version 2.0 (the "License");
  4. // you may not use this file except in compliance with the License.
  5. // You may obtain a copy of the License at
  6. //
  7. // http://www.apache.org/licenses/LICENSE-2.0
  8. //
  9. // Unless required by applicable law or agreed to in writing, software
  10. // distributed under the License is distributed on an "AS IS" BASIS,
  11. // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  12. // See the License for the specific language governing permissions and
  13. // limitations under the License.
  14. package com.google.enterprise.connector.servlet;
  15. import com.google.enterprise.connector.manager.Context;
  16. import java.net.InetAddress;
  17. import java.net.UnknownHostException;
  18. import java.util.Properties;
  19. import java.util.logging.Level;
  20. import java.util.logging.Logger;
  21. /**
  22. * Remote Address Filter that can be used to restrict access to the
  23. * Servlets. In general, we allow access from the GSA, localhost,
  24. * proxies, and perhaps certain adminstrator machines.
  25. */
  26. // TODO: This was intended to be extensible replacement for the
  27. // Tomcat RemoteIPAddrValve. It has since been stripped of all
  28. // such pretense and becomes, instead, a more generalized version
  29. // of the hack perpetrated in revision r2607.
  30. // Offboard Connector Managers must still rely on Tomcat RemoteIPAddrValve
  31. // to determine access.
  32. // TODO: This should really be a javax.servlet.Filter.
  33. public class RemoteAddressFilter {
  34. private static Logger LOGGER =
  35. Logger.getLogger(ServletUtil.class.getName());
  36. // Singleton instance.
  37. private static RemoteAddressFilter instance = null;
  38. // If true, Connector Manager is on-board GSA.
  39. private final boolean onboard;
  40. /**
  41. * The various access modes. The GSA has access to all the servlets.
  42. * The access mode names here are arbitrary, as their interpretations
  43. * vary greatly between GSA on-board and off-board.
  44. */
  45. public enum Access {
  46. RED, BLACK
  47. }
  48. /** Restrict constructor to singlton. */
  49. private RemoteAddressFilter() {
  50. Properties props = Context.getInstance().getConnectorManagerProperties();
  51. this.onboard = Boolean.valueOf(props.getProperty("manager.onboard"));
  52. // TODO: extract accept/deny IP address patterns from properties.
  53. LOGGER.config(((this.onboard) ? "On-board" : "External")
  54. + " Connector Manager detected.");
  55. }
  56. /** Returns the singleton RemoteAddressFilter for this context. */
  57. public static synchronized RemoteAddressFilter getInstance() {
  58. // Delay instantiating the singleton until we are sure the
  59. // Context has been initialized.
  60. if (instance == null) {
  61. instance = new RemoteAddressFilter();
  62. }
  63. return instance;
  64. }
  65. /**
  66. * Determine whether the supplied {@code remoteAddr} is allowed
  67. * access under the given {@code Access} mode.
  68. *
  69. * @param accessMode the {@code Access} mode for the calling servlet.
  70. * @param remoteAddr the IP address of the servlet's caller.
  71. * @return {@code true} if the caller is permitted access to the servlet,
  72. * {@code false} otherwise.
  73. */
  74. public boolean allowed(Access accessMode, String remoteAddr) {
  75. // Offboard - defer to RemoteIPAddr Valve.
  76. // Onboard - RED is Public access, BLACK is GSA or localhost only.
  77. if (!onboard || accessMode == Access.RED) {
  78. return true;
  79. }
  80. try {
  81. InetAddress caller = InetAddress.getByName(remoteAddr);
  82. if (caller.isLoopbackAddress() ||
  83. caller.equals(InetAddress.getLocalHost())) {
  84. return true; // localhost is allowed access
  85. }
  86. String gsaHost = Context.getInstance().getGsaFeedHost();
  87. InetAddress[] gsaAddrs = InetAddress.getAllByName(gsaHost);
  88. for (int i = 0; i < gsaAddrs.length; i++) {
  89. if (caller.equals(gsaAddrs[i])) {
  90. return true; // GSA is allowed access
  91. }
  92. }
  93. LOGGER.warning("Denying caller: " + caller );
  94. } catch (UnknownHostException uhe) {
  95. // Unknown host - fall through to fail.
  96. LOGGER.log(Level.WARNING, "Denying caller:" + remoteAddr, uhe);
  97. }
  98. return false;
  99. }
  100. /**
  101. * Determine whether the supplied {@code remoteAddr} is the configured
  102. * Feed Host.
  103. *
  104. * @param remoteAddr the IP address of the servlet's caller.
  105. * @return {@code true} if the caller is the Feed Host,
  106. * {@code false} otherwise.
  107. */
  108. public boolean isFeedHost(String remoteAddr) {
  109. try {
  110. InetAddress caller = InetAddress.getByName(remoteAddr);
  111. if (onboard) {
  112. // If onboard, localhost is feed host.
  113. return caller.isLoopbackAddress() ||
  114. caller.equals(InetAddress.getLocalHost());
  115. }
  116. String gsaHost = Context.getInstance().getGsaFeedHost();
  117. InetAddress[] gsaAddrs = InetAddress.getAllByName(gsaHost);
  118. for (int i = 0; i < gsaAddrs.length; i++) {
  119. if (caller.equals(gsaAddrs[i])) {
  120. return true; // The GSA Feed Host is the caller.
  121. }
  122. }
  123. } catch (UnknownHostException uhe) {
  124. // Unknown host - fall through to fail.
  125. }
  126. return false;
  127. }
  128. }