Add an abusive message filter interface and submodule
This commit is contained in:
		
							parent
							
								
									ae7f8af03e
								
							
						
					
					
						commit
						ad1aeea74b
					
				| 
						 | 
					@ -0,0 +1,11 @@
 | 
				
			||||||
 | 
					# Note that the implmentation of the abusive message filter is private; internal
 | 
				
			||||||
 | 
					# developers will need to override this URL with:
 | 
				
			||||||
 | 
					#
 | 
				
			||||||
 | 
					# ```
 | 
				
			||||||
 | 
					# git config submodule.abusive-message-filter.url PRIVATE_URL
 | 
				
			||||||
 | 
					# ```
 | 
				
			||||||
 | 
					#
 | 
				
			||||||
 | 
					# External developers may safely ignore this submodule.
 | 
				
			||||||
 | 
					[submodule "abusive-message-filter"]
 | 
				
			||||||
 | 
						path = abusive-message-filter
 | 
				
			||||||
 | 
						url = REDACTED
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1 @@
 | 
				
			||||||
 | 
					Subproject commit 817f2e867ec1160f793c95f16f50b75c35f2e887
 | 
				
			||||||
							
								
								
									
										23
									
								
								pom.xml
								
								
								
								
							
							
						
						
									
										23
									
								
								pom.xml
								
								
								
								
							| 
						 | 
					@ -298,6 +298,29 @@
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  </dependencies>
 | 
					  </dependencies>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  <profiles>
 | 
				
			||||||
 | 
					    <profile>
 | 
				
			||||||
 | 
					      <id>include-abusive-message-filter</id>
 | 
				
			||||||
 | 
					      <activation>
 | 
				
			||||||
 | 
					        <file>
 | 
				
			||||||
 | 
					          <exists>abusive-message-filter/pom.xml</exists>
 | 
				
			||||||
 | 
					        </file>
 | 
				
			||||||
 | 
					      </activation>
 | 
				
			||||||
 | 
					      <modules>
 | 
				
			||||||
 | 
					        <module>abusive-message-filter</module>
 | 
				
			||||||
 | 
					      </modules>
 | 
				
			||||||
 | 
					    </profile>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    <profile>
 | 
				
			||||||
 | 
					      <id>exclude-abusive-message-filter</id>
 | 
				
			||||||
 | 
					      <activation>
 | 
				
			||||||
 | 
					        <file>
 | 
				
			||||||
 | 
					          <missing>abusive-message-filter/pom.xml</missing>
 | 
				
			||||||
 | 
					        </file>
 | 
				
			||||||
 | 
					      </activation>
 | 
				
			||||||
 | 
					    </profile>
 | 
				
			||||||
 | 
					  </profiles>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  <build>
 | 
					  <build>
 | 
				
			||||||
    <extensions>
 | 
					    <extensions>
 | 
				
			||||||
      <extension>
 | 
					      <extension>
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
							
								
								
									
										208
									
								
								service/pom.xml
								
								
								
								
							
							
						
						
									
										208
									
								
								service/pom.xml
								
								
								
								
							| 
						 | 
					@ -462,104 +462,112 @@
 | 
				
			||||||
    </dependency>
 | 
					    </dependency>
 | 
				
			||||||
  </dependencies>
 | 
					  </dependencies>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  <profiles>
 | 
				
			||||||
 | 
					    <profile>
 | 
				
			||||||
 | 
					      <id>exclude-abusive-message-filter</id>
 | 
				
			||||||
 | 
					      <build>
 | 
				
			||||||
 | 
					        <plugins>
 | 
				
			||||||
 | 
					          <plugin>
 | 
				
			||||||
 | 
					            <groupId>org.apache.maven.plugins</groupId>
 | 
				
			||||||
 | 
					            <artifactId>maven-shade-plugin</artifactId>
 | 
				
			||||||
 | 
					            <version>3.2.4</version>
 | 
				
			||||||
 | 
					            <configuration>
 | 
				
			||||||
 | 
					              <createDependencyReducedPom>true</createDependencyReducedPom>
 | 
				
			||||||
 | 
					              <filters>
 | 
				
			||||||
 | 
					                <filter>
 | 
				
			||||||
 | 
					                  <artifact>*:*</artifact>
 | 
				
			||||||
 | 
					                  <excludes>
 | 
				
			||||||
 | 
					                    <exclude>META-INF/*.SF</exclude>
 | 
				
			||||||
 | 
					                    <exclude>META-INF/*.DSA</exclude>
 | 
				
			||||||
 | 
					                    <exclude>META-INF/*.RSA</exclude>
 | 
				
			||||||
 | 
					                  </excludes>
 | 
				
			||||||
 | 
					                </filter>
 | 
				
			||||||
 | 
					              </filters>
 | 
				
			||||||
 | 
					            </configuration>
 | 
				
			||||||
 | 
					            <executions>
 | 
				
			||||||
 | 
					              <execution>
 | 
				
			||||||
 | 
					                <phase>package</phase>
 | 
				
			||||||
 | 
					                <goals>
 | 
				
			||||||
 | 
					                  <goal>shade</goal>
 | 
				
			||||||
 | 
					                </goals>
 | 
				
			||||||
 | 
					                <configuration>
 | 
				
			||||||
 | 
					                  <transformers>
 | 
				
			||||||
 | 
					                    <transformer implementation="org.apache.maven.plugins.shade.resource.ServicesResourceTransformer"/>
 | 
				
			||||||
 | 
					                    <transformer implementation="org.apache.maven.plugins.shade.resource.ManifestResourceTransformer">
 | 
				
			||||||
 | 
					                      <mainClass>org.whispersystems.textsecuregcm.WhisperServerService</mainClass>
 | 
				
			||||||
 | 
					                    </transformer>
 | 
				
			||||||
 | 
					                  </transformers>
 | 
				
			||||||
 | 
					                </configuration>
 | 
				
			||||||
 | 
					              </execution>
 | 
				
			||||||
 | 
					            </executions>
 | 
				
			||||||
 | 
					          </plugin>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					          <plugin>
 | 
				
			||||||
 | 
					            <groupId>org.apache.maven.plugins</groupId>
 | 
				
			||||||
 | 
					            <artifactId>maven-assembly-plugin</artifactId>
 | 
				
			||||||
 | 
					            <version>3.3.0</version>
 | 
				
			||||||
 | 
					            <configuration>
 | 
				
			||||||
 | 
					              <descriptors>
 | 
				
			||||||
 | 
					                <descriptor>assembly.xml</descriptor>
 | 
				
			||||||
 | 
					              </descriptors>
 | 
				
			||||||
 | 
					            </configuration>
 | 
				
			||||||
 | 
					            <executions>
 | 
				
			||||||
 | 
					              <execution>
 | 
				
			||||||
 | 
					                <id>make-assembly</id> <!-- this is used for inheritance merges -->
 | 
				
			||||||
 | 
					                <phase>package</phase> <!-- bind to the packaging phase -->
 | 
				
			||||||
 | 
					                <goals>
 | 
				
			||||||
 | 
					                  <goal>single</goal>
 | 
				
			||||||
 | 
					                </goals>
 | 
				
			||||||
 | 
					              </execution>
 | 
				
			||||||
 | 
					            </executions>
 | 
				
			||||||
 | 
					          </plugin>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					          <plugin>
 | 
				
			||||||
 | 
					            <groupId>org.codehaus.mojo</groupId>
 | 
				
			||||||
 | 
					            <artifactId>properties-maven-plugin</artifactId>
 | 
				
			||||||
 | 
					            <version>1.0.0</version>
 | 
				
			||||||
 | 
					            <executions>
 | 
				
			||||||
 | 
					              <execution>
 | 
				
			||||||
 | 
					                <id>read-deploy-configuration</id>
 | 
				
			||||||
 | 
					                <phase>deploy</phase>
 | 
				
			||||||
 | 
					                <goals>
 | 
				
			||||||
 | 
					                  <goal>read-project-properties</goal>
 | 
				
			||||||
 | 
					                </goals>
 | 
				
			||||||
 | 
					                <configuration>
 | 
				
			||||||
 | 
					                  <files>${project.basedir}/config/deploy.properties</files>
 | 
				
			||||||
 | 
					                </configuration>
 | 
				
			||||||
 | 
					              </execution>
 | 
				
			||||||
 | 
					            </executions>
 | 
				
			||||||
 | 
					          </plugin>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					          <plugin>
 | 
				
			||||||
 | 
					            <groupId>org.signal</groupId>
 | 
				
			||||||
 | 
					            <artifactId>s3-upload-maven-plugin</artifactId>
 | 
				
			||||||
 | 
					            <version>1.6-SNAPSHOT</version>
 | 
				
			||||||
 | 
					            <configuration>
 | 
				
			||||||
 | 
					              <source>${project.build.directory}/${project.build.finalName}-bin.tar.gz</source>
 | 
				
			||||||
 | 
					              <bucketName>${deploy.bucketName}</bucketName>
 | 
				
			||||||
 | 
					              <region>${deploy.bucketRegion}</region>
 | 
				
			||||||
 | 
					              <destination>${project.build.finalName}-bin.tar.gz</destination>
 | 
				
			||||||
 | 
					            </configuration>
 | 
				
			||||||
 | 
					            <executions>
 | 
				
			||||||
 | 
					              <execution>
 | 
				
			||||||
 | 
					                <id>deploy-to-s3</id>
 | 
				
			||||||
 | 
					                <phase>deploy</phase>
 | 
				
			||||||
 | 
					                <goals>
 | 
				
			||||||
 | 
					                  <goal>s3-upload</goal>
 | 
				
			||||||
 | 
					                </goals>
 | 
				
			||||||
 | 
					              </execution>
 | 
				
			||||||
 | 
					            </executions>
 | 
				
			||||||
 | 
					          </plugin>
 | 
				
			||||||
 | 
					        </plugins>
 | 
				
			||||||
 | 
					      </build>
 | 
				
			||||||
 | 
					    </profile>
 | 
				
			||||||
 | 
					  </profiles>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  <build>
 | 
					  <build>
 | 
				
			||||||
    <finalName>${project.parent.artifactId}-${project.version}</finalName>
 | 
					    <finalName>${project.parent.artifactId}-${project.version}</finalName>
 | 
				
			||||||
    <plugins>
 | 
					    <plugins>
 | 
				
			||||||
      <plugin>
 | 
					 | 
				
			||||||
        <groupId>org.apache.maven.plugins</groupId>
 | 
					 | 
				
			||||||
        <artifactId>maven-shade-plugin</artifactId>
 | 
					 | 
				
			||||||
        <version>3.2.4</version>
 | 
					 | 
				
			||||||
        <configuration>
 | 
					 | 
				
			||||||
          <createDependencyReducedPom>true</createDependencyReducedPom>
 | 
					 | 
				
			||||||
          <filters>
 | 
					 | 
				
			||||||
            <filter>
 | 
					 | 
				
			||||||
              <artifact>*:*</artifact>
 | 
					 | 
				
			||||||
              <excludes>
 | 
					 | 
				
			||||||
                <exclude>META-INF/*.SF</exclude>
 | 
					 | 
				
			||||||
                <exclude>META-INF/*.DSA</exclude>
 | 
					 | 
				
			||||||
                <exclude>META-INF/*.RSA</exclude>
 | 
					 | 
				
			||||||
              </excludes>
 | 
					 | 
				
			||||||
            </filter>
 | 
					 | 
				
			||||||
          </filters>
 | 
					 | 
				
			||||||
        </configuration>
 | 
					 | 
				
			||||||
        <executions>
 | 
					 | 
				
			||||||
          <execution>
 | 
					 | 
				
			||||||
            <phase>package</phase>
 | 
					 | 
				
			||||||
            <goals>
 | 
					 | 
				
			||||||
              <goal>shade</goal>
 | 
					 | 
				
			||||||
            </goals>
 | 
					 | 
				
			||||||
            <configuration>
 | 
					 | 
				
			||||||
              <transformers>
 | 
					 | 
				
			||||||
                <transformer implementation="org.apache.maven.plugins.shade.resource.ServicesResourceTransformer"/>
 | 
					 | 
				
			||||||
                <transformer implementation="org.apache.maven.plugins.shade.resource.ManifestResourceTransformer">
 | 
					 | 
				
			||||||
                  <mainClass>org.whispersystems.textsecuregcm.WhisperServerService</mainClass>
 | 
					 | 
				
			||||||
                </transformer>
 | 
					 | 
				
			||||||
              </transformers>
 | 
					 | 
				
			||||||
            </configuration>
 | 
					 | 
				
			||||||
          </execution>
 | 
					 | 
				
			||||||
        </executions>
 | 
					 | 
				
			||||||
      </plugin>
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
      <plugin>
 | 
					 | 
				
			||||||
        <groupId>org.apache.maven.plugins</groupId>
 | 
					 | 
				
			||||||
        <artifactId>maven-assembly-plugin</artifactId>
 | 
					 | 
				
			||||||
        <version>3.3.0</version>
 | 
					 | 
				
			||||||
        <configuration>
 | 
					 | 
				
			||||||
          <descriptors>
 | 
					 | 
				
			||||||
            <descriptor>assembly.xml</descriptor>
 | 
					 | 
				
			||||||
          </descriptors>
 | 
					 | 
				
			||||||
        </configuration>
 | 
					 | 
				
			||||||
        <executions>
 | 
					 | 
				
			||||||
          <execution>
 | 
					 | 
				
			||||||
            <id>make-assembly</id> <!-- this is used for inheritance merges -->
 | 
					 | 
				
			||||||
            <phase>package</phase> <!-- bind to the packaging phase -->
 | 
					 | 
				
			||||||
            <goals>
 | 
					 | 
				
			||||||
              <goal>single</goal>
 | 
					 | 
				
			||||||
            </goals>
 | 
					 | 
				
			||||||
          </execution>
 | 
					 | 
				
			||||||
        </executions>
 | 
					 | 
				
			||||||
      </plugin>
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
      <plugin>
 | 
					 | 
				
			||||||
        <groupId>org.codehaus.mojo</groupId>
 | 
					 | 
				
			||||||
        <artifactId>properties-maven-plugin</artifactId>
 | 
					 | 
				
			||||||
        <version>1.0.0</version>
 | 
					 | 
				
			||||||
        <executions>
 | 
					 | 
				
			||||||
          <execution>
 | 
					 | 
				
			||||||
            <id>read-deploy-configuration</id>
 | 
					 | 
				
			||||||
            <phase>deploy</phase>
 | 
					 | 
				
			||||||
            <goals>
 | 
					 | 
				
			||||||
              <goal>read-project-properties</goal>
 | 
					 | 
				
			||||||
            </goals>
 | 
					 | 
				
			||||||
            <configuration>
 | 
					 | 
				
			||||||
              <files>${project.basedir}/config/deploy.properties</files>
 | 
					 | 
				
			||||||
            </configuration>
 | 
					 | 
				
			||||||
          </execution>
 | 
					 | 
				
			||||||
        </executions>
 | 
					 | 
				
			||||||
      </plugin>
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
      <plugin>
 | 
					 | 
				
			||||||
        <groupId>org.signal</groupId>
 | 
					 | 
				
			||||||
        <artifactId>s3-upload-maven-plugin</artifactId>
 | 
					 | 
				
			||||||
        <version>1.6-SNAPSHOT</version>
 | 
					 | 
				
			||||||
        <configuration>
 | 
					 | 
				
			||||||
          <source>${project.build.directory}/${project.build.finalName}-bin.tar.gz</source>
 | 
					 | 
				
			||||||
          <bucketName>${deploy.bucketName}</bucketName>
 | 
					 | 
				
			||||||
          <region>${deploy.bucketRegion}</region>
 | 
					 | 
				
			||||||
          <destination>${project.build.finalName}-bin.tar.gz</destination>
 | 
					 | 
				
			||||||
        </configuration>
 | 
					 | 
				
			||||||
        <executions>
 | 
					 | 
				
			||||||
          <execution>
 | 
					 | 
				
			||||||
            <id>deploy-to-s3</id>
 | 
					 | 
				
			||||||
            <phase>deploy</phase>
 | 
					 | 
				
			||||||
            <goals>
 | 
					 | 
				
			||||||
              <goal>s3-upload</goal>
 | 
					 | 
				
			||||||
            </goals>
 | 
					 | 
				
			||||||
          </execution>
 | 
					 | 
				
			||||||
        </executions>
 | 
					 | 
				
			||||||
      </plugin>
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
      <plugin>
 | 
					      <plugin>
 | 
				
			||||||
        <groupId>org.codehaus.mojo</groupId>
 | 
					        <groupId>org.codehaus.mojo</groupId>
 | 
				
			||||||
        <artifactId>templating-maven-plugin</artifactId>
 | 
					        <artifactId>templating-maven-plugin</artifactId>
 | 
				
			||||||
| 
						 | 
					@ -573,6 +581,18 @@
 | 
				
			||||||
          </execution>
 | 
					          </execution>
 | 
				
			||||||
        </executions>
 | 
					        </executions>
 | 
				
			||||||
      </plugin>
 | 
					      </plugin>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      <plugin>
 | 
				
			||||||
 | 
					        <groupId>org.apache.maven.plugins</groupId>
 | 
				
			||||||
 | 
					        <artifactId>maven-jar-plugin</artifactId>
 | 
				
			||||||
 | 
					        <executions>
 | 
				
			||||||
 | 
					          <execution>
 | 
				
			||||||
 | 
					            <goals>
 | 
				
			||||||
 | 
					              <goal>test-jar</goal>
 | 
				
			||||||
 | 
					            </goals>
 | 
				
			||||||
 | 
					          </execution>
 | 
				
			||||||
 | 
					        </executions>
 | 
				
			||||||
 | 
					      </plugin>
 | 
				
			||||||
    </plugins>
 | 
					    </plugins>
 | 
				
			||||||
  </build>
 | 
					  </build>
 | 
				
			||||||
</project>
 | 
					</project>
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -13,6 +13,7 @@ import java.util.List;
 | 
				
			||||||
import java.util.Map;
 | 
					import java.util.Map;
 | 
				
			||||||
import javax.validation.Valid;
 | 
					import javax.validation.Valid;
 | 
				
			||||||
import javax.validation.constraints.NotNull;
 | 
					import javax.validation.constraints.NotNull;
 | 
				
			||||||
 | 
					import org.whispersystems.textsecuregcm.configuration.AbusiveMessageFilterConfiguration;
 | 
				
			||||||
import org.whispersystems.textsecuregcm.configuration.AccountDatabaseCrawlerConfiguration;
 | 
					import org.whispersystems.textsecuregcm.configuration.AccountDatabaseCrawlerConfiguration;
 | 
				
			||||||
import org.whispersystems.textsecuregcm.configuration.AccountsDatabaseConfiguration;
 | 
					import org.whispersystems.textsecuregcm.configuration.AccountsDatabaseConfiguration;
 | 
				
			||||||
import org.whispersystems.textsecuregcm.configuration.AccountsDynamoDbConfiguration;
 | 
					import org.whispersystems.textsecuregcm.configuration.AccountsDynamoDbConfiguration;
 | 
				
			||||||
| 
						 | 
					@ -330,6 +331,10 @@ public class WhisperServerConfiguration extends Configuration {
 | 
				
			||||||
  @JsonProperty
 | 
					  @JsonProperty
 | 
				
			||||||
  private ReportMessageConfiguration reportMessage = new ReportMessageConfiguration();
 | 
					  private ReportMessageConfiguration reportMessage = new ReportMessageConfiguration();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  @Valid
 | 
				
			||||||
 | 
					  @JsonProperty
 | 
				
			||||||
 | 
					  private AbusiveMessageFilterConfiguration abusiveMessageFilter;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  private Map<String, String> transparentDataIndex = new HashMap<>();
 | 
					  private Map<String, String> transparentDataIndex = new HashMap<>();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  public StripeConfiguration getStripe() {
 | 
					  public StripeConfiguration getStripe() {
 | 
				
			||||||
| 
						 | 
					@ -565,4 +570,8 @@ public class WhisperServerConfiguration extends Configuration {
 | 
				
			||||||
  public ReportMessageConfiguration getReportMessageConfiguration() {
 | 
					  public ReportMessageConfiguration getReportMessageConfiguration() {
 | 
				
			||||||
    return reportMessage;
 | 
					    return reportMessage;
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  public AbusiveMessageFilterConfiguration getAbusiveMessageFilterConfiguration() {
 | 
				
			||||||
 | 
					    return abusiveMessageFilter;
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -44,6 +44,7 @@ import java.util.Collections;
 | 
				
			||||||
import java.util.EnumSet;
 | 
					import java.util.EnumSet;
 | 
				
			||||||
import java.util.List;
 | 
					import java.util.List;
 | 
				
			||||||
import java.util.Optional;
 | 
					import java.util.Optional;
 | 
				
			||||||
 | 
					import java.util.ServiceLoader;
 | 
				
			||||||
import java.util.concurrent.ArrayBlockingQueue;
 | 
					import java.util.concurrent.ArrayBlockingQueue;
 | 
				
			||||||
import java.util.concurrent.BlockingQueue;
 | 
					import java.util.concurrent.BlockingQueue;
 | 
				
			||||||
import java.util.concurrent.ExecutorService;
 | 
					import java.util.concurrent.ExecutorService;
 | 
				
			||||||
| 
						 | 
					@ -52,6 +53,7 @@ import java.util.concurrent.TimeUnit;
 | 
				
			||||||
import javax.servlet.DispatcherType;
 | 
					import javax.servlet.DispatcherType;
 | 
				
			||||||
import javax.servlet.FilterRegistration;
 | 
					import javax.servlet.FilterRegistration;
 | 
				
			||||||
import javax.servlet.ServletRegistration;
 | 
					import javax.servlet.ServletRegistration;
 | 
				
			||||||
 | 
					import javax.ws.rs.container.DynamicFeature;
 | 
				
			||||||
import org.eclipse.jetty.servlets.CrossOriginFilter;
 | 
					import org.eclipse.jetty.servlets.CrossOriginFilter;
 | 
				
			||||||
import org.glassfish.jersey.server.ServerProperties;
 | 
					import org.glassfish.jersey.server.ServerProperties;
 | 
				
			||||||
import org.jdbi.v3.core.Jdbi;
 | 
					import org.jdbi.v3.core.Jdbi;
 | 
				
			||||||
| 
						 | 
					@ -64,6 +66,8 @@ import org.signal.zkgroup.receipts.ServerZkReceiptOperations;
 | 
				
			||||||
import org.slf4j.Logger;
 | 
					import org.slf4j.Logger;
 | 
				
			||||||
import org.slf4j.LoggerFactory;
 | 
					import org.slf4j.LoggerFactory;
 | 
				
			||||||
import org.whispersystems.dispatch.DispatchManager;
 | 
					import org.whispersystems.dispatch.DispatchManager;
 | 
				
			||||||
 | 
					import org.whispersystems.textsecuregcm.abuse.AbusiveMessageFilter;
 | 
				
			||||||
 | 
					import org.whispersystems.textsecuregcm.abuse.FilterAbusiveMessages;
 | 
				
			||||||
import org.whispersystems.textsecuregcm.auth.AccountAuthenticator;
 | 
					import org.whispersystems.textsecuregcm.auth.AccountAuthenticator;
 | 
				
			||||||
import org.whispersystems.textsecuregcm.auth.AuthenticatedAccount;
 | 
					import org.whispersystems.textsecuregcm.auth.AuthenticatedAccount;
 | 
				
			||||||
import org.whispersystems.textsecuregcm.auth.CertificateGenerator;
 | 
					import org.whispersystems.textsecuregcm.auth.CertificateGenerator;
 | 
				
			||||||
| 
						 | 
					@ -659,6 +663,32 @@ public class WhisperServerService extends Application<WhisperServerConfiguration
 | 
				
			||||||
      webSocketEnvironment.jersey().register(controller);
 | 
					      webSocketEnvironment.jersey().register(controller);
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    boolean registeredAbusiveMessageFilter = false;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    for (final AbusiveMessageFilter filter : ServiceLoader.load(AbusiveMessageFilter.class)) {
 | 
				
			||||||
 | 
					      if (filter.getClass().isAnnotationPresent(FilterAbusiveMessages.class)) {
 | 
				
			||||||
 | 
					        try {
 | 
				
			||||||
 | 
					          filter.configure(config.getAbusiveMessageFilterConfiguration().getEnvironment());
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					          environment.lifecycle().manage(filter);
 | 
				
			||||||
 | 
					          environment.jersey().register(filter);
 | 
				
			||||||
 | 
					          webSocketEnvironment.jersey().register(filter);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					          log.info("Registered abusive message filter: {}", filter.getClass().getName());
 | 
				
			||||||
 | 
					          registeredAbusiveMessageFilter = true;
 | 
				
			||||||
 | 
					        } catch (final Exception e) {
 | 
				
			||||||
 | 
					          log.warn("Failed to register abusive message filter: {}", filter.getClass().getName(), e);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					      } else {
 | 
				
			||||||
 | 
					        log.warn("Abusive message filter {} not annotated with @FilterAbusiveMessages and will not be installed",
 | 
				
			||||||
 | 
					            filter.getClass().getName());
 | 
				
			||||||
 | 
					      }
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if (!registeredAbusiveMessageFilter) {
 | 
				
			||||||
 | 
					      log.warn("No abusive message filters installed");
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    WebSocketEnvironment<AuthenticatedAccount> provisioningEnvironment = new WebSocketEnvironment<>(environment,
 | 
					    WebSocketEnvironment<AuthenticatedAccount> provisioningEnvironment = new WebSocketEnvironment<>(environment,
 | 
				
			||||||
        webSocketEnvironment.getRequestLog(), 60000);
 | 
					        webSocketEnvironment.getRequestLog(), 60000);
 | 
				
			||||||
    provisioningEnvironment.jersey().register(new WebsocketRefreshApplicationEventListener(accountsManager, clientPresenceManager));
 | 
					    provisioningEnvironment.jersey().register(new WebsocketRefreshApplicationEventListener(accountsManager, clientPresenceManager));
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,33 @@
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * Copyright 2013-2021 Signal Messenger, LLC
 | 
				
			||||||
 | 
					 * SPDX-License-Identifier: AGPL-3.0-only
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					package org.whispersystems.textsecuregcm.abuse;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import io.dropwizard.lifecycle.Managed;
 | 
				
			||||||
 | 
					import javax.ws.rs.container.ContainerRequestFilter;
 | 
				
			||||||
 | 
					import java.io.IOException;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/**
 | 
				
			||||||
 | 
					 * An abusive message filter is a {@link ContainerRequestFilter} that filters requests to message-sending endpoints to
 | 
				
			||||||
 | 
					 * detect and respond to patterns of abusive behavior.
 | 
				
			||||||
 | 
					 * <p/>
 | 
				
			||||||
 | 
					 * Abusive message filters are managed components that are generally loaded dynamically via a
 | 
				
			||||||
 | 
					 * {@link java.util.ServiceLoader}. Their {@link #configure(String)} method will be called prior to be adding to the
 | 
				
			||||||
 | 
					 * server's pool of {@link Managed} objects.
 | 
				
			||||||
 | 
					 * <p/>
 | 
				
			||||||
 | 
					 * Abusive message filters must be annotated with {@link FilterAbusiveMessages}, a name binding annotation that
 | 
				
			||||||
 | 
					 * restricts the endpoints to which the filter may apply.
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					public interface AbusiveMessageFilter extends ContainerRequestFilter, Managed {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  /**
 | 
				
			||||||
 | 
					   * Configures this abusive message filter. This method will be called before the filter is added to the server's pool
 | 
				
			||||||
 | 
					   * of managed objects and before the server processes any requests.
 | 
				
			||||||
 | 
					   *
 | 
				
			||||||
 | 
					   * @param environmentName the name of the environment in which this filter is running (e.g. "staging" or "production")
 | 
				
			||||||
 | 
					   * @throws IOException if the filter could not read its configuration source for any reason
 | 
				
			||||||
 | 
					   */
 | 
				
			||||||
 | 
					  void configure(String environmentName) throws IOException;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,21 @@
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * Copyright 2013-2021 Signal Messenger, LLC
 | 
				
			||||||
 | 
					 * SPDX-License-Identifier: AGPL-3.0-only
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					package org.whispersystems.textsecuregcm.abuse;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import javax.ws.rs.NameBinding;
 | 
				
			||||||
 | 
					import java.lang.annotation.ElementType;
 | 
				
			||||||
 | 
					import java.lang.annotation.Retention;
 | 
				
			||||||
 | 
					import java.lang.annotation.RetentionPolicy;
 | 
				
			||||||
 | 
					import java.lang.annotation.Target;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/**
 | 
				
			||||||
 | 
					 * A name-binding annotation that associates {@link AbusiveMessageFilter}s with resource methods.
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					@NameBinding
 | 
				
			||||||
 | 
					@Retention(RetentionPolicy.RUNTIME)
 | 
				
			||||||
 | 
					@Target({ElementType.TYPE, ElementType.METHOD})
 | 
				
			||||||
 | 
					public @interface FilterAbusiveMessages {
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,26 @@
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * Copyright 2013-2021 Signal Messenger, LLC
 | 
				
			||||||
 | 
					 * SPDX-License-Identifier: AGPL-3.0-only
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					package org.whispersystems.textsecuregcm.configuration;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import com.fasterxml.jackson.annotation.JsonCreator;
 | 
				
			||||||
 | 
					import com.fasterxml.jackson.annotation.JsonProperty;
 | 
				
			||||||
 | 
					import javax.validation.constraints.NotBlank;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					public class AbusiveMessageFilterConfiguration {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  @JsonProperty
 | 
				
			||||||
 | 
					  @NotBlank
 | 
				
			||||||
 | 
					  private final String environment;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  @JsonCreator
 | 
				
			||||||
 | 
					  public AbusiveMessageFilterConfiguration(@JsonProperty("environment") final String environment) {
 | 
				
			||||||
 | 
					    this.environment = environment;
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  public String getEnvironment() {
 | 
				
			||||||
 | 
					    return environment;
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
| 
						 | 
					@ -58,6 +58,7 @@ import javax.ws.rs.core.Response;
 | 
				
			||||||
import javax.ws.rs.core.Response.Status;
 | 
					import javax.ws.rs.core.Response.Status;
 | 
				
			||||||
import org.slf4j.Logger;
 | 
					import org.slf4j.Logger;
 | 
				
			||||||
import org.slf4j.LoggerFactory;
 | 
					import org.slf4j.LoggerFactory;
 | 
				
			||||||
 | 
					import org.whispersystems.textsecuregcm.abuse.FilterAbusiveMessages;
 | 
				
			||||||
import org.whispersystems.textsecuregcm.auth.Anonymous;
 | 
					import org.whispersystems.textsecuregcm.auth.Anonymous;
 | 
				
			||||||
import org.whispersystems.textsecuregcm.auth.AuthenticatedAccount;
 | 
					import org.whispersystems.textsecuregcm.auth.AuthenticatedAccount;
 | 
				
			||||||
import org.whispersystems.textsecuregcm.auth.CombinedUnidentifiedSenderAccessKeys;
 | 
					import org.whispersystems.textsecuregcm.auth.CombinedUnidentifiedSenderAccessKeys;
 | 
				
			||||||
| 
						 | 
					@ -163,6 +164,7 @@ public class MessageController {
 | 
				
			||||||
  @PUT
 | 
					  @PUT
 | 
				
			||||||
  @Consumes(MediaType.APPLICATION_JSON)
 | 
					  @Consumes(MediaType.APPLICATION_JSON)
 | 
				
			||||||
  @Produces(MediaType.APPLICATION_JSON)
 | 
					  @Produces(MediaType.APPLICATION_JSON)
 | 
				
			||||||
 | 
					  @FilterAbusiveMessages
 | 
				
			||||||
  public Response sendMessage(@Auth Optional<AuthenticatedAccount> source,
 | 
					  public Response sendMessage(@Auth Optional<AuthenticatedAccount> source,
 | 
				
			||||||
      @HeaderParam(OptionalAccess.UNIDENTIFIED) Optional<Anonymous> accessKey,
 | 
					      @HeaderParam(OptionalAccess.UNIDENTIFIED) Optional<Anonymous> accessKey,
 | 
				
			||||||
      @HeaderParam("User-Agent") String userAgent,
 | 
					      @HeaderParam("User-Agent") String userAgent,
 | 
				
			||||||
| 
						 | 
					@ -285,6 +287,7 @@ public class MessageController {
 | 
				
			||||||
  @PUT
 | 
					  @PUT
 | 
				
			||||||
  @Consumes(MultiRecipientMessageProvider.MEDIA_TYPE)
 | 
					  @Consumes(MultiRecipientMessageProvider.MEDIA_TYPE)
 | 
				
			||||||
  @Produces(MediaType.APPLICATION_JSON)
 | 
					  @Produces(MediaType.APPLICATION_JSON)
 | 
				
			||||||
 | 
					  @FilterAbusiveMessages
 | 
				
			||||||
  public Response sendMultiRecipientMessage(
 | 
					  public Response sendMultiRecipientMessage(
 | 
				
			||||||
      @HeaderParam(OptionalAccess.UNIDENTIFIED) CombinedUnidentifiedSenderAccessKeys accessKeys,
 | 
					      @HeaderParam(OptionalAccess.UNIDENTIFIED) CombinedUnidentifiedSenderAccessKeys accessKeys,
 | 
				
			||||||
      @HeaderParam("User-Agent") String userAgent,
 | 
					      @HeaderParam("User-Agent") String userAgent,
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -59,7 +59,7 @@ public class DynamicConfigurationManager<T> {
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  @VisibleForTesting
 | 
					  @VisibleForTesting
 | 
				
			||||||
  public DynamicConfigurationManager(AppConfigClient appConfigClient, String application, String environment,
 | 
					  DynamicConfigurationManager(AppConfigClient appConfigClient, String application, String environment,
 | 
				
			||||||
      String configurationName, String clientId, Class<T> configurationClass) {
 | 
					      String configurationName, String clientId, Class<T> configurationClass) {
 | 
				
			||||||
    this.appConfigClient = appConfigClient;
 | 
					    this.appConfigClient = appConfigClient;
 | 
				
			||||||
    this.application = application;
 | 
					    this.application = application;
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
		Loading…
	
		Reference in New Issue